Subversion Repositories SmartDukaan

Rev

Rev 36040 | Details | Compare with Previous | Last modification | View Log | RSS feed

Rev Author Line No. Line
24417 govind 1
package com.spice.profitmandi.web.controller;
2
 
3
import com.spice.profitmandi.common.exception.ProfitMandiBusinessException;
4
import com.spice.profitmandi.common.model.CustomRetailer;
32812 shampa 5
import com.spice.profitmandi.common.model.ProfitMandiConstants;
24417 govind 6
import com.spice.profitmandi.dao.entity.auth.AuthUser;
27690 amit.gupta 7
import com.spice.profitmandi.dao.entity.cs.*;
36040 ranu 8
import com.spice.profitmandi.dao.entity.cs.TicketReadStatus.UserType;
27270 tejbeer 9
import com.spice.profitmandi.dao.entity.dtr.Document;
24417 govind 10
import com.spice.profitmandi.dao.entity.fofo.ActivityType;
11
import com.spice.profitmandi.dao.enumuration.cs.EscalationType;
24699 govind 12
import com.spice.profitmandi.dao.enumuration.cs.TicketStatus;
25570 tejbeer 13
import com.spice.profitmandi.dao.model.CreatePositionModel;
24417 govind 14
import com.spice.profitmandi.dao.repository.auth.AuthRepository;
27690 amit.gupta 15
import com.spice.profitmandi.dao.repository.cs.*;
27270 tejbeer 16
import com.spice.profitmandi.dao.repository.dtr.DocumentRepository;
25570 tejbeer 17
import com.spice.profitmandi.dao.repository.dtr.FofoStoreRepository;
24417 govind 18
import com.spice.profitmandi.service.authentication.RoleManager;
36040 ranu 19
import com.spice.profitmandi.service.mail.MailOutboxService;
24417 govind 20
import com.spice.profitmandi.service.user.RetailerService;
21
import com.spice.profitmandi.web.model.LoginDetails;
22
import com.spice.profitmandi.web.util.CookiesProcessor;
23
import com.spice.profitmandi.web.util.MVCResponseSender;
27690 amit.gupta 24
import org.apache.logging.log4j.LogManager;
25
import org.apache.logging.log4j.Logger;
26
import org.springframework.beans.factory.annotation.Autowired;
27
import org.springframework.mail.javamail.JavaMailSender;
28
import org.springframework.stereotype.Controller;
29
import org.springframework.transaction.annotation.Transactional;
30
import org.springframework.ui.Model;
31
import org.springframework.web.bind.annotation.*;
24417 govind 32
 
27690 amit.gupta 33
import javax.servlet.http.HttpServletRequest;
36040 ranu 34
import javax.swing.*;
27690 amit.gupta 35
import java.time.LocalDateTime;
36
import java.util.*;
37
import java.util.stream.Collectors;
38
 
24417 govind 39
@Controller
40
@Transactional(rollbackFor = Throwable.class)
41
public class CsController {
42
 
31762 tejbeer 43
    private static final Logger LOGGER = LogManager.getLogger(CsController.class);
44
    private static final String ACTIVITY_SUBJECT = "Message related ticketId#%s";
45
    private static final String PARTNER_RESOLVED_TICKET_MAIL = "Dear Partner , we have resolved your ticket # %s , request to kindly accept the same. In case you still have any concerns regarding the same pls click on %s so that we can help you.Regards\nSmartdukaan";
46
    private static final String PARTNER_REOPEN = "Dear Partner , Your ticket # %s has been re-opened as per your confirmation & we are committed to resolve it on priority.Regards\nSmartdukaan";
47
    private static final String INTERNAL_REOPEN_MAIL = "Team, Pls note that the Ticket Id %s has been re-opened by %s , pls respond on priority";
48
    private static final String INTERNAL_REOPEN_ACTIVITY_MESSAGE = "Hi,My ticket is not resolved yet,so I have reopened it";
24699 govind 49
 
31762 tejbeer 50
    @Autowired
51
    JavaMailSender mailSender;
24620 govind 52
 
31762 tejbeer 53
    @Autowired
35957 amit 54
    MailOutboxService mailOutboxService;
55
 
56
    @Autowired
31762 tejbeer 57
    private CsService csService;
24417 govind 58
 
31762 tejbeer 59
    @Autowired
60
    private CookiesProcessor cookiesProcessor;
24417 govind 61
 
31762 tejbeer 62
    @Autowired
63
    private TicketCategoryRepository ticketCategoryRepository;
24417 govind 64
 
31762 tejbeer 65
    @Autowired
66
    private TicketSubCategoryRepository ticketSubCategoryRepository;
24417 govind 67
 
31762 tejbeer 68
    @Autowired
69
    private RegionRepository regionRepository;
24417 govind 70
 
31762 tejbeer 71
    @Autowired
72
    private RetailerService retailerService;
24417 govind 73
 
31762 tejbeer 74
    @Autowired
75
    private MVCResponseSender mvcResponseSender;
24417 govind 76
 
31762 tejbeer 77
    @Autowired
78
    private AuthRepository authRepository;
24417 govind 79
 
31762 tejbeer 80
    @Autowired
81
    private PositionRepository positionRepository;
24417 govind 82
 
31762 tejbeer 83
    @Autowired
84
    private TicketRepository ticketRepository;
24417 govind 85
 
31762 tejbeer 86
    @Autowired
87
    private RoleManager roleManager;
24417 govind 88
 
31762 tejbeer 89
    @Autowired
90
    private ActivityRepository activityRepository;
24417 govind 91
 
31762 tejbeer 92
    @Autowired
93
    private ActivityAttachmentRepository activityAttachmentRepository;
27270 tejbeer 94
 
31762 tejbeer 95
    @Autowired
96
    private TicketAssignedRepository ticketAssignedRepository;
24569 govind 97
 
31762 tejbeer 98
    @Autowired
99
    private PartnerRegionRepository partnerRegionRepository;
24500 govind 100
 
31762 tejbeer 101
    @Autowired
32493 amit.gupta 102
    PartnerPositionRepository partnerPositionRepository;
25570 tejbeer 103
 
31762 tejbeer 104
    @Autowired
105
    FofoStoreRepository fofoStoreRepository;
25570 tejbeer 106
 
31762 tejbeer 107
    @Autowired
108
    DocumentRepository documentRepository;
27270 tejbeer 109
 
31762 tejbeer 110
    @GetMapping(value = "/cs/createCategory")
111
    public String getCreateCategory(HttpServletRequest request, Model model) {
112
        List<TicketCategory> ticketCategories = ticketCategoryRepository.selectAll();
113
        model.addAttribute("ticketCategories", ticketCategories);
114
        return "create-ticket-category";
115
    }
24417 govind 116
 
31762 tejbeer 117
    @PostMapping(value = "/cs/createCategory")
33081 ranu 118
    public String createCategory(HttpServletRequest request,
119
                                 @RequestParam(name = "name") String name,
120
                                 @RequestParam(name = "categoryType") int categoryType,
121
                                 @RequestParam(name = "description") String description,
122
                                 Model model) throws ProfitMandiBusinessException {
31762 tejbeer 123
        TicketCategory ticketCategory = ticketCategoryRepository.selectByName(name);
124
        if (ticketCategory != null) {
125
            throw new ProfitMandiBusinessException("name", name, "already exists!");
126
        }
33081 ranu 127
 
31762 tejbeer 128
        ticketCategory = new TicketCategory();
129
        ticketCategory.setName(name);
130
        ticketCategory.setDescription(description);
33081 ranu 131
 
132
        ticketCategory.setCategoryType(categoryType == 1);
31762 tejbeer 133
        ticketCategoryRepository.persist(ticketCategory);
134
        return "create-ticket-category";
135
    }
24417 govind 136
 
33081 ranu 137
 
31762 tejbeer 138
    @GetMapping(value = "/cs/createSubCategory")
139
    public String getCreateSubCategory(HttpServletRequest request, Model model) {
140
        List<TicketCategory> ticketCategories = ticketCategoryRepository.selectAll();
141
        model.addAttribute("ticketCategories", ticketCategories);
142
        return "create-ticket-sub-category";
143
    }
24417 govind 144
 
31762 tejbeer 145
    @GetMapping(value = "/cs/getSubCategoryByCategoryId")
146
    public String getSubCategoryByCategoryId(HttpServletRequest request, @RequestParam(name = "ticketCategoryId", defaultValue = "") int ticketCategoryId, Model model) {
147
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAll(ticketCategoryId);
148
        TicketCategory ticketCategory = ticketCategoryRepository.selectById(ticketCategoryId);
33081 ranu 149
        LOGGER.info("ticketSubCategories {}", ticketSubCategories);
150
        LOGGER.info("ticketCategory {}", ticketCategory);
31762 tejbeer 151
        model.addAttribute("ticketSubCategories", ticketSubCategories);
152
        model.addAttribute("ticketCategory", ticketCategory);
153
        return "ticket-sub-category";
154
    }
24417 govind 155
 
31762 tejbeer 156
    @PostMapping(value = "/cs/createSubCategory")
157
    public String createSubCategory(HttpServletRequest request, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "name") String name, @RequestParam(name = "description") String description, Model model) throws ProfitMandiBusinessException {
24417 govind 158
 
31762 tejbeer 159
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectTicketSubCategory(categoryId, name);
160
        if (ticketSubCategory != null) {
161
            throw new ProfitMandiBusinessException("name & categoryId", name + "  " + categoryId, "already exists!");
162
        }
24417 govind 163
 
31762 tejbeer 164
        ticketSubCategory = new TicketSubCategory();
165
        ticketSubCategory.setCategoryId(categoryId);
166
        ticketSubCategory.setName(name);
167
        ticketSubCategory.setDescription(description);
168
        ticketSubCategoryRepository.persist(ticketSubCategory);
169
        return "create-ticket-sub-category";
170
    }
24417 govind 171
 
31762 tejbeer 172
    @GetMapping(value = "/cs/createRegion")
173
    public String createRegion(HttpServletRequest request, Model model) {
174
        List<Region> regions = regionRepository.selectAll();
175
        model.addAttribute("regions", regions);
176
        return "create-region";
177
    }
24417 govind 178
 
31762 tejbeer 179
    @PostMapping(value = "/cs/createRegion")
180
    public String createRegion(HttpServletRequest request, @RequestParam(name = "name") String name, @RequestParam(name = "description") String description, Model model) throws Exception {
181
        Region region = regionRepository.selectByName(name);
182
        if (region != null) {
183
            throw new ProfitMandiBusinessException("name", name, "already exists!");
184
        }
185
        region = new Region();
186
        region.setName(name);
187
        region.setDescription(description);
188
        regionRepository.persist(region);
189
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
190
        return "response";
191
    }
24417 govind 192
 
31762 tejbeer 193
    @GetMapping(value = "/cs/getPartners")
33244 ranu 194
    public String getPartners(HttpServletRequest request, @RequestParam(name = "regionId", defaultValue = "0") int regionId, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 195
        List<Integer> fofoIds = fofoStoreRepository.selectAll().stream().map(x -> x.getId()).collect(Collectors.toList());
196
        List<Integer> addedfofoIds = partnerRegionRepository.selectByRegionId(regionId).stream().map(x -> x.getFofoId()).collect(Collectors.toList());
30426 tejbeer 197
 
31762 tejbeer 198
        Map<Integer, CustomRetailer> customRetailerMap = retailerService.getAllFofoRetailers();
199
        Map<Integer, CustomRetailer> fofoRetailers = fofoIds.stream().map(x -> customRetailerMap.get(x)).filter(x -> x != null).collect(Collectors.toList()).stream().collect(Collectors.toMap(x -> x.getPartnerId(), x -> x));
200
        model.addAttribute("fofoRetailers", fofoRetailers);
201
        model.addAttribute("addedfofoIds", addedfofoIds);
202
        return "added-region-partners";
203
    }
24569 govind 204
 
31762 tejbeer 205
    @GetMapping(value = "/cs/getPartnersByRegion")
33244 ranu 206
    public String getPartnersByRegion(HttpServletRequest request, @RequestParam(name = "regionId", defaultValue = "0") int regionId, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 207
        List<Integer> fofoIds = null;
208
        fofoIds = partnerRegionRepository.selectByRegionId(regionId).stream().map(x -> x.getFofoId()).collect(Collectors.toList());
25570 tejbeer 209
 
31762 tejbeer 210
        if (fofoIds.contains(0)) {
211
            fofoIds = fofoStoreRepository.selectAll().stream().filter(x -> x.isActive()).collect(Collectors.toList()).stream().map(x -> x.getId()).collect(Collectors.toList());
25570 tejbeer 212
 
31762 tejbeer 213
        }
214
        Map<Integer, CustomRetailer> customRetailerMap = retailerService.getAllFofoRetailers();
25570 tejbeer 215
 
31762 tejbeer 216
        Map<Integer, CustomRetailer> fofoRetailers = fofoIds.stream().map(x -> customRetailerMap.get(x)).filter(x -> x != null).collect(Collectors.toList()).stream().collect(Collectors.toMap(x -> x.getPartnerId(), x -> x));
217
        model.addAttribute("fofoRetailers", fofoRetailers);
218
        return "added-subregion-partners";
219
    }
25570 tejbeer 220
 
31762 tejbeer 221
    @GetMapping(value = "/cs/createPartnerRegion")
222
    public String createPartnerRegion(HttpServletRequest request, Model model) {
223
        List<Region> regions = regionRepository.selectAll();
224
        model.addAttribute("regions", regions);
225
        return "create-partner-region";
226
    }
24417 govind 227
 
31762 tejbeer 228
    @PostMapping(value = "/cs/createPartnerRegion")
229
    public String createPartnerRegion(HttpServletRequest request, @RequestParam(name = "regionId") int regionId, @RequestBody List<Integer> selectedFofoIds, Model model) throws Exception {
230
        partnerRegionRepository.delete(regionId);
231
        LOGGER.info("successfully removed");
232
        LOGGER.info(selectedFofoIds.size());
233
        csService.addPartnerToRegion(regionId, selectedFofoIds);
234
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
235
        return "response";
236
    }
24417 govind 237
 
31762 tejbeer 238
    @GetMapping(value = "/cs/getPosition")
33244 ranu 239
    public String getPosition(HttpServletRequest request, @RequestParam int positionId, Model model) throws ProfitMandiBusinessException {
27410 tejbeer 240
 
31762 tejbeer 241
        Position position = positionRepository.selectById(positionId);
27410 tejbeer 242
 
31762 tejbeer 243
        List<CustomRetailer> positionIdCustomRetailer = csService.getPositionCustomRetailerMap(Arrays.asList(position)).get(position.getId());
27410 tejbeer 244
 
31762 tejbeer 245
        Map<Integer, CustomRetailer> regionRetailerMap = csService.getRegionPartners(Arrays.asList(position)).get(position.getRegionId()).stream().collect(Collectors.toMap(x -> x.getPartnerId(), x -> x));
27410 tejbeer 246
 
31762 tejbeer 247
        model.addAttribute("position", position);
248
        model.addAttribute("regionRetailerMap", regionRetailerMap);
249
        model.addAttribute("positionIdCustomRetailer", positionIdCustomRetailer);
27410 tejbeer 250
 
31762 tejbeer 251
        return "position-partner";
252
    }
27410 tejbeer 253
 
31762 tejbeer 254
    @GetMapping(value = "/cs/createPosition")
255
    public String createPosition(HttpServletRequest request, @RequestParam(name = "offset", defaultValue = "0") int offset, @RequestParam(name = "limit", defaultValue = "10") int limit, Model model) {
256
        List<AuthUser> authUsers = authRepository.selectAllActiveUser();
257
        List<TicketCategory> ticketCategories = ticketCategoryRepository.selectAll();
258
        List<Region> regions = regionRepository.selectAll();
259
        model.addAttribute("escalationTypes", EscalationType.values());
260
        model.addAttribute("authUsers", authUsers);
261
        model.addAttribute("ticketCategories", ticketCategories);
262
        model.addAttribute("regions", regions);
24500 govind 263
 
31762 tejbeer 264
        List<Position> positions = positionRepository.selectAllPosition();
265
        LOGGER.info("positions" + positions);
30426 tejbeer 266
 
31762 tejbeer 267
        Map<Integer, AuthUser> authUserIdAndAuthUserMap = csService.getAuthUserIdAndAuthUserMapUsingPositions(positions);
268
        Map<Integer, TicketCategory> categoryIdAndCategoryMap = csService.getCategoryIdAndCategoryUsingPositions(positions);
269
        Map<Integer, Region> regionIdAndRegionMap = csService.getRegionIdAndRegionMap(positions);
25570 tejbeer 270
 
27410 tejbeer 271
//	    Map<Integer, List<CustomRetailer>> positionIdAndpartnerRegionMap = csService
272
//			.getpositionIdAndpartnerRegionMap(positions);
25570 tejbeer 273
 
27410 tejbeer 274
//	     Map<Integer, List<CustomRetailer>> addedpositionIdAndCustomRetailerMap = csService
275
//				.getPositionCustomRetailerMap(positions);
276
//		LOGGER.info("fofoIdAndCustomRetailerMap" + addedpositionIdAndCustomRetailerMap);
24500 govind 277
 
31762 tejbeer 278
        model.addAttribute("start", offset + 1);
30426 tejbeer 279
 
31762 tejbeer 280
        model.addAttribute("positions", positions);
281
        model.addAttribute("authUserIdAndAuthUserMap", authUserIdAndAuthUserMap);
282
        model.addAttribute("categoryIdAndCategoryMap", categoryIdAndCategoryMap);
283
        model.addAttribute("regionIdAndRegionMap", regionIdAndRegionMap);
284
        // model.addAttribute("positionIdAndCustomRetailerMap",
285
        // addedpositionIdAndCustomRetailerMap);
286
        // model.addAttribute("positionIdAndpartnerRegionMap",
27410 tejbeer 287
// positionIdAndpartnerRegionMap);
25570 tejbeer 288
 
31762 tejbeer 289
        return "create-position";
290
    }
24500 govind 291
 
31762 tejbeer 292
    @GetMapping(value = "/cs/position-paginated")
293
    public String positionPaginated(HttpServletRequest request, @RequestParam(name = "offset", defaultValue = "0") int offset, @RequestParam(name = "limit", defaultValue = "10") int limit, Model model) {
24500 govind 294
 
31762 tejbeer 295
        List<Position> positions = positionRepository.selectAll(offset, limit);
296
        Map<Integer, AuthUser> authUserIdAndAuthUserMap = csService.getAuthUserIdAndAuthUserMapUsingPositions(positions);
297
        Map<Integer, TicketCategory> categoryIdAndCategoryMap = csService.getCategoryIdAndCategoryUsingPositions(positions);
298
        Map<Integer, Region> regionIdAndRegionMap = csService.getRegionIdAndRegionMap(positions);
299
        /*
300
         * Map<Integer, List<CustomRetailer>> positionIdAndpartnerRegionMap = csService
301
         * .getpositionIdAndpartnerRegionMap(positions);
302
         * 
303
         * Map<Integer, List<CustomRetailer>> addedpositionIdAndCustomRetailerMap =
304
         * csService .getPositionCustomRetailerMap(positions);
305
         */
25570 tejbeer 306
 
31762 tejbeer 307
        model.addAttribute("positions", positions);
308
        model.addAttribute("authUserIdAndAuthUserMap", authUserIdAndAuthUserMap);
309
        model.addAttribute("categoryIdAndCategoryMap", categoryIdAndCategoryMap);
310
        model.addAttribute("regionIdAndRegionMap", regionIdAndRegionMap);
311
        // model.addAttribute("positionIdAndCustomRetailerMap",
312
        // addedpositionIdAndCustomRetailerMap);
313
        // model.addAttribute("positionIdAndpartnerRegionMap",
314
        // positionIdAndpartnerRegionMap);
25570 tejbeer 315
 
31762 tejbeer 316
        return "position-paginated";
317
    }
24417 govind 318
 
31762 tejbeer 319
    @PostMapping(value = "/cs/createPosition")
320
    public String createPosition(HttpServletRequest request, @RequestBody CreatePositionModel createPositionModel, Model model) throws Exception {
24417 govind 321
 
35570 amit 322
        LOGGER.info("partnerPosition" + createPositionModel.isTicketAssigned());
25570 tejbeer 323
 
35570 amit 324
        // Validate authUserId exists
325
        int authUserId = createPositionModel.getAuthUserId();
326
        if (authUserId <= 0 || authRepository.selectById(authUserId) == null) {
327
            throw new ProfitMandiBusinessException("Position", authUserId, "Invalid authUserId");
328
        }
329
 
330
        // Validate categoryId exists (0 means "All Categories")
331
        int categoryId = createPositionModel.getCategoryId();
332
        if (categoryId > 0 && ticketCategoryRepository.selectById(categoryId) == null) {
333
            throw new ProfitMandiBusinessException("Position", categoryId, "Invalid categoryId");
334
        }
335
 
336
        // Validate regionId exists (0 or 5 typically means "All Partners/Regions")
337
        int regionId = createPositionModel.getRegionId();
338
        if (regionId > 0 && regionId != 5 && regionRepository.selectById(regionId) == null) {
339
            throw new ProfitMandiBusinessException("Position", regionId, "Invalid regionId");
340
        }
341
 
35589 amit 342
        // Validate fofoIds exist (0 means "All Partners")
35570 amit 343
        List<Integer> fofoIds = createPositionModel.getFofoIds();
35590 amit 344
        if (fofoIds == null || fofoIds.isEmpty()) {
345
            throw new ProfitMandiBusinessException("Position", 0, "At least one partner must be specified");
346
        }
347
        boolean hasAllPartners = fofoIds.contains(0);
348
        if (hasAllPartners && fofoIds.size() > 1) {
349
            throw new ProfitMandiBusinessException("Position", 0, "Cannot mix 'All Partners' (0) with specific partner IDs");
350
        }
351
        if (!hasAllPartners) {
352
            Map<Integer, CustomRetailer> validRetailers = retailerService.getFofoRetailers(false);
353
            for (int fofoId : fofoIds) {
354
                if (!validRetailers.containsKey(fofoId)) {
355
                    throw new ProfitMandiBusinessException("Position", fofoId, "Invalid fofoId");
35570 amit 356
                }
357
            }
358
        }
359
 
360
        Position position = positionRepository.selectPosition(authUserId, categoryId, regionId, createPositionModel.getEscalationType());
31762 tejbeer 361
        if (position == null) {
362
            position = new Position();
35570 amit 363
            position.setAuthUserId(authUserId);
364
            position.setCategoryId(categoryId);
31762 tejbeer 365
            position.setEscalationType(createPositionModel.getEscalationType());
35570 amit 366
            position.setRegionId(regionId);
31762 tejbeer 367
            position.setCreateTimestamp(LocalDateTime.now());
368
            position.setTicketAssignee(createPositionModel.isTicketAssigned());
369
            positionRepository.persist(position);
25570 tejbeer 370
 
35570 amit 371
            if (fofoIds != null) {
372
                for (int fofoId : fofoIds) {
373
                    PartnerPosition partnerPosition = new PartnerPosition();
374
                    partnerPosition.setFofoId(fofoId);
375
                    partnerPosition.setRegionId(regionId);
376
                    partnerPosition.setPositionId(position.getId());
377
                    partnerPositionRepository.persist(partnerPosition);
378
                    LOGGER.info("partnerPosition" + partnerPosition);
379
                }
31762 tejbeer 380
            }
24417 govind 381
 
31762 tejbeer 382
            model.addAttribute("response1", mvcResponseSender.createResponseString(true));
383
        } else {
35570 amit 384
            throw new ProfitMandiBusinessException("Position", authUserId, "already exists!");
31762 tejbeer 385
        }
386
        return "response";
387
    }
25570 tejbeer 388
 
31762 tejbeer 389
    @PostMapping(value = "/cs/updatePartnerPosition")
390
    public String updatePartnerPosition(HttpServletRequest request, @RequestParam(name = "regionId") int regionId, @RequestBody List<Integer> selectedFofoIds, @RequestParam(name = "positionId") int positionId, Model model) throws Exception {
25570 tejbeer 391
 
35571 amit 392
        // Validate positionId exists
393
        Position position = positionRepository.selectById(positionId);
394
        if (position == null) {
395
            throw new ProfitMandiBusinessException("Position", positionId, "Position not found");
396
        }
397
 
398
        // Validate regionId exists (0 or 5 typically means "All Partners/Regions")
399
        if (regionId > 0 && regionId != 5 && regionRepository.selectById(regionId) == null) {
400
            throw new ProfitMandiBusinessException("Position", regionId, "Invalid regionId");
401
        }
402
 
35589 amit 403
        // Validate fofoIds exist (0 means "All Partners")
35590 amit 404
        if (selectedFofoIds == null || selectedFofoIds.isEmpty()) {
405
            throw new ProfitMandiBusinessException("Position", 0, "At least one partner must be specified");
406
        }
407
        boolean hasAllPartners = selectedFofoIds.contains(0);
408
        if (hasAllPartners && selectedFofoIds.size() > 1) {
409
            throw new ProfitMandiBusinessException("Position", 0, "Cannot mix 'All Partners' (0) with specific partner IDs");
410
        }
411
        if (!hasAllPartners) {
35607 amit 412
            // Use direct database query instead of cached retailerService to include newly added partners
413
            Set<Integer> validFofoIds = fofoStoreRepository.selectAll().stream()
414
                    .map(fs -> fs.getId())
415
                    .collect(Collectors.toSet());
35590 amit 416
            for (int fofoId : selectedFofoIds) {
35607 amit 417
                if (!validFofoIds.contains(fofoId)) {
35590 amit 418
                    throw new ProfitMandiBusinessException("Position", fofoId, "Invalid fofoId");
35571 amit 419
                }
420
            }
421
        }
422
 
32493 amit.gupta 423
        partnerPositionRepository.delete(positionId);
35571 amit 424
        if (selectedFofoIds != null) {
425
            for (int fofoId : selectedFofoIds) {
426
                PartnerPosition partnerPosition = new PartnerPosition();
427
                partnerPosition.setFofoId(fofoId);
428
                partnerPosition.setRegionId(regionId);
429
                partnerPosition.setPositionId(positionId);
430
                partnerPositionRepository.persist(partnerPosition);
431
            }
31762 tejbeer 432
        }
25570 tejbeer 433
 
31762 tejbeer 434
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
25570 tejbeer 435
 
31762 tejbeer 436
        return "response";
437
    }
24417 govind 438
 
31762 tejbeer 439
    @GetMapping(value = "/cs/createTicket")
440
    public String createTicket(HttpServletRequest request, Model model) throws ProfitMandiBusinessException {
441
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
35594 amit 442
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
443
        boolean isCrmUser = false;
444
 
31762 tejbeer 445
        List<TicketCategory> ticketCategories = csService.getAllTicketCategotyFromSubCategory();
35594 amit 446
 
447
        if (isAdmin) {
448
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
449
            isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
450
 
451
            // CRM users should not see Sales and RBM categories
452
            if (isCrmUser) {
453
                ticketCategories = ticketCategories.stream()
454
                        .filter(tc -> tc.getId() != ProfitMandiConstants.TICKET_CATEGORY_SALES
455
                                && tc.getId() != ProfitMandiConstants.TICKET_CATEGORY_RBM)
456
                        .collect(Collectors.toList());
457
            }
35626 amit 458
        } else {
459
            // For partners/customers: hide categories where ALL subcategories are invisible
460
            ticketCategories = ticketCategories.stream()
461
                    .filter(category -> {
462
                        List<TicketSubCategory> subs = ticketSubCategoryRepository.selectAll(category.getId());
463
                        // Keep category if at least one subcategory is visible
464
                        return subs.stream().anyMatch(TicketSubCategory::isVisibility);
465
                    })
466
                    .collect(Collectors.toList());
35594 amit 467
        }
468
 
469
        model.addAttribute("roleType", isAdmin);
31762 tejbeer 470
        model.addAttribute("ticketCategories", ticketCategories);
35594 amit 471
        model.addAttribute("isCrmUser", isCrmUser);
31762 tejbeer 472
        return "create-ticket";
473
    }
24417 govind 474
 
31762 tejbeer 475
    @GetMapping(value = "/cs/getSubCategoriesByCategoryId")
35594 amit 476
    public String getSubCategoriesByCategoryId(HttpServletRequest request, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, Model model) throws ProfitMandiBusinessException {
477
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
478
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
479
        boolean isCrmUser = false;
480
 
481
        if (isAdmin) {
482
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
483
            isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
484
        }
485
 
31830 amit.gupta 486
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAllVisible(categoryId);
35594 amit 487
 
35617 amit 488
        // CRM users should not see invisible subcategories
35594 amit 489
        if (isCrmUser) {
490
            ticketSubCategories = ticketSubCategories.stream()
35617 amit 491
                    .filter(TicketSubCategory::isVisibility)
35594 amit 492
                    .collect(Collectors.toList());
493
        }
494
 
31762 tejbeer 495
        LOGGER.info(ticketSubCategories);
496
        model.addAttribute("ticketSubCategories", ticketSubCategories);
497
        return "ticket-sub-categories";
498
    }
24417 govind 499
 
24791 govind 500
 
31762 tejbeer 501
    @GetMapping(value = "/cs/getEscalationTypeByCategoryId")
502
    public String getEscalationTypeByCategoryId(HttpServletRequest request, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "authId", defaultValue = "0") int authId, Model model) {
503
        List<Position> positions = positionRepository.selectPositionbyCategoryIdAndAuthId(categoryId, authId);
504
        List<EscalationType> escalationTypes = new ArrayList<>();
24791 govind 505
 
31762 tejbeer 506
        if (!positions.isEmpty()) {
507
            escalationTypes = positions.stream().map(x -> x.getEscalationType()).distinct().collect(Collectors.toList());
508
        }
24620 govind 509
 
31762 tejbeer 510
        LOGGER.info("escalationTypes {}", escalationTypes);
24500 govind 511
 
31762 tejbeer 512
        model.addAttribute("escalationTypes", escalationTypes);
513
        return "ticket-escalationtype";
514
    }
24500 govind 515
 
24824 govind 516
 
31762 tejbeer 517
    @GetMapping(value = "/cs/getCategoriesByAuthId")
518
    public String getCategoriesByAuthId(HttpServletRequest request, @RequestParam(name = "authId", defaultValue = "0") int authId, Model model) {
24824 govind 519
 
24787 govind 520
 
31762 tejbeer 521
        List<Position> positions = positionRepository.selectPositionByAuthId(authId);
24791 govind 522
 
31762 tejbeer 523
        LOGGER.info("positions {}", positions);
24417 govind 524
 
31762 tejbeer 525
        List<TicketCategory> ticketCategories = new ArrayList<TicketCategory>();
24417 govind 526
 
31762 tejbeer 527
        if (!positions.isEmpty()) {
24417 govind 528
 
31762 tejbeer 529
            List<Integer> categoryIds = positions.stream().map(x -> x.getCategoryId()).collect(Collectors.toList());
530
            ticketCategories = ticketCategoryRepository.selectAll(categoryIds);
531
        }
532
        LOGGER.info("ticketCategories {}", ticketCategories);
533
        model.addAttribute("ticketCategories", ticketCategories);
534
        return "ticket-categories";
535
    }
27270 tejbeer 536
 
31762 tejbeer 537
    @PostMapping(value = "/cs/createTicket")
538
    public String createTicket(HttpServletRequest request, @RequestParam(name = "categoryId") int categoryId, @RequestParam(name = "subCategoryId") int subCategoryId, @RequestParam(name = "message") String message, Model model) throws Exception {
539
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
540
        List<Ticket> tickets = ticketRepository.selectAllResolvedMarkedTicketByCreator(loginDetails.getFofoId());
541
        if (tickets.size() > 3 || tickets.size() == 3) {
542
            model.addAttribute("response1", mvcResponseSender.createResponseString(false));
543
        } else {
544
            csService.createTicket(loginDetails.getFofoId(), categoryId, subCategoryId, message);
545
            model.addAttribute("response1", mvcResponseSender.createResponseString(true));
546
        }
547
        return "response";
548
    }
27270 tejbeer 549
 
31762 tejbeer 550
    @GetMapping(value = "/cs/myticket")
35626 amit 551
    public String getTicket(HttpServletRequest request,
552
                            @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
553
                            @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
554
                            @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
555
                            @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
556
                            @RequestParam(name = "page", defaultValue = "0") int page,
557
                            @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
558
                            @RequestParam(name = "search", required = false) String searchText,
559
                            Model model) throws ProfitMandiBusinessException {
560
        populateMyTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
561
        return "ticket";
562
    }
563
 
564
    @GetMapping(value = "/cs/myticket-content")
565
    public String getTicketContent(HttpServletRequest request,
566
                                   @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
567
                                   @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
568
                                   @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
569
                                   @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
570
                                   @RequestParam(name = "page", defaultValue = "0") int page,
571
                                   @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
572
                                   @RequestParam(name = "search", required = false) String searchText,
573
                                   Model model) throws ProfitMandiBusinessException {
574
        populateMyTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
575
        return "ticket-content";
576
    }
577
 
578
    private void populateMyTicketModel(HttpServletRequest request, SortOrder sortOrder, TicketStatus ticketStatus,
579
                                       TicketSearchType ticketSearchType, int searchTerm, int page, int pageSize,
580
                                       String searchText, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 581
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
582
        List<Ticket> tickets = null;
583
        List<TicketAssigned> ticketAssigneds = null;
35626 amit 584
        long totalRecords = 0;
31762 tejbeer 585
        Map<Integer, AuthUser> authUserIdAndAuthUserMap = null;
35569 amit 586
        boolean isAdmin = roleManager.isAdmin(new HashSet<>(loginDetails.getRoleIds()));
587
        AuthUser currentAuthUser = isAdmin ? authRepository.selectByEmailOrMobile(loginDetails.getEmailId()) : null;
588
 
35592 amit 589
        boolean isCrmUser = isAdmin && currentAuthUser != null && positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
35603 amit 590
        boolean isSalesUser = isAdmin && currentAuthUser != null && positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_SALES);
591
        boolean isRbmUser = isAdmin && currentAuthUser != null && positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_RBM);
35592 amit 592
 
35626 amit 593
        List<Integer> userCategoryIds = isAdmin && currentAuthUser != null
594
                ? positionRepository.selectCategoryIdsByAuthUserId(currentAuthUser.getId())
595
                : Collections.emptyList();
596
 
597
        int offset = page * pageSize;
598
 
35569 amit 599
        if (isAdmin) {
600
            int authUserId = currentAuthUser.getId();
35592 amit 601
 
602
            if (isCrmUser) {
35626 amit 603
                Optional<Boolean> resolvedOpt = ticketStatus.equals(TicketStatus.RESOLVED) ? Optional.empty() : Optional.of(TicketStatus.CLOSED.equals(ticketStatus));
604
                tickets = ticketRepository.selectAllTicketsPaginated(
605
                        resolvedOpt, sortOrder, ticketSearchType, searchTerm, searchText,
606
                        userCategoryIds, offset, pageSize);
607
                totalRecords = ticketRepository.selectAllTicketsPaginatedCount(
608
                        resolvedOpt, ticketSearchType, searchTerm, searchText, userCategoryIds);
31762 tejbeer 609
            } else {
35626 amit 610
                Optional<Boolean> resolvedOpt = ticketStatus.equals(TicketStatus.RESOLVED) ? Optional.empty() : Optional.of(TicketStatus.CLOSED.equals(ticketStatus));
611
                tickets = ticketRepository.selectAllByAssigneePaginated(
612
                        authUserId, resolvedOpt, sortOrder, ticketSearchType, searchTerm, searchText,
613
                        offset, pageSize);
614
                totalRecords = ticketRepository.selectAllByAssigneePaginatedCount(
615
                        authUserId, resolvedOpt, ticketSearchType, searchTerm, searchText);
31762 tejbeer 616
            }
35626 amit 617
 
31762 tejbeer 618
            if (tickets.size() > 0) {
619
                ticketAssigneds = ticketAssignedRepository.selectByTicketIds(tickets.stream().map(x -> x.getId()).collect(Collectors.toList()));
620
                authUserIdAndAuthUserMap = csService.getAuthUserIdAndAuthUserMap(ticketAssigneds);
621
                Map<Integer, CustomRetailer> fofoIdsAndCustomRetailer = csService.getPartnerByFofoIds(tickets);
622
                model.addAttribute("fofoIdsAndCustomRetailer", fofoIdsAndCustomRetailer);
623
            }
24500 govind 624
 
31762 tejbeer 625
        } else {
626
            tickets = ticketRepository.selectAllByCreator(loginDetails.getFofoId(), Optional.of(TicketStatus.OPENED.equals(ticketStatus)), sortOrder);
35626 amit 627
            totalRecords = ticketRepository.selectAllCountByCreator(loginDetails.getFofoId(), Optional.of(TicketStatus.OPENED.equals(ticketStatus)));
31762 tejbeer 628
        }
629
        authUserIdAndAuthUserMap = csService.getTicketIdAndAuthUserMapUsingTickets(tickets);
35571 amit 630
        if (authUserIdAndAuthUserMap == null) {
631
            authUserIdAndAuthUserMap = new HashMap<>();
632
        }
24620 govind 633
 
36040 ranu 634
        Map<Integer, List<AuthUser>> authUserListMap = csService.getAssignedAuthList(tickets);
635
        if (authUserListMap == null) {
636
            authUserListMap = new HashMap<>();
637
        }
638
 
35626 amit 639
        int totalPages = (int) Math.ceil((double) totalRecords / pageSize);
640
        if (totalPages == 0) totalPages = 1;
641
        int startRecord = totalRecords > 0 ? offset + 1 : 0;
642
        int endRecord = (int) Math.min(offset + pageSize, totalRecords);
643
 
644
        model.addAttribute("size", totalRecords);
645
        model.addAttribute("totalRecords", totalRecords);
646
        model.addAttribute("currentPage", page);
647
        model.addAttribute("pageSize", pageSize);
648
        model.addAttribute("totalPages", totalPages);
649
        model.addAttribute("startRecord", startRecord);
650
        model.addAttribute("endRecord", endRecord);
651
        model.addAttribute("searchText", searchText != null ? searchText : "");
652
        model.addAttribute("currentPageDisplay", page + 1);
653
        model.addAttribute("prevPage", page - 1);
654
        model.addAttribute("nextPage", page + 1);
655
        model.addAttribute("lastPage", totalPages - 1);
656
 
35569 amit 657
        model.addAttribute("roleType", isAdmin);
658
        model.addAttribute("isCrmUser", isCrmUser);
35603 amit 659
        model.addAttribute("isSalesUser", isSalesUser);
660
        model.addAttribute("salesCategoryId", ProfitMandiConstants.TICKET_CATEGORY_SALES);
661
        model.addAttribute("isRbmUser", isRbmUser);
662
        model.addAttribute("rbmCategoryId", ProfitMandiConstants.TICKET_CATEGORY_RBM);
35626 amit 663
        model.addAttribute("userCategoryIds", userCategoryIds);
24500 govind 664
 
31762 tejbeer 665
        List<Integer> subCategoryIds = tickets.stream().map(x -> x.getSubCategoryId()).collect(Collectors.toList());
666
        Map<Integer, TicketSubCategory> subCategoryIdAndSubCategoryMap = csService.getSubCategoryIdAndSubCategoryMap(subCategoryIds);
35571 amit 667
        if (subCategoryIdAndSubCategoryMap == null) {
668
            subCategoryIdAndSubCategoryMap = new HashMap<>();
669
        }
24747 govind 670
 
31762 tejbeer 671
        Map<Integer, TicketCategory> subCategoryIdAndCategoryMap = csService.getSubCategoryIdAndCategoryMap(subCategoryIds);
35571 amit 672
        if (subCategoryIdAndCategoryMap == null) {
673
            subCategoryIdAndCategoryMap = new HashMap<>();
674
        }
27318 amit.gupta 675
 
31762 tejbeer 676
        List<Integer> ticketIds = tickets.stream().map(x -> x.getId()).collect(Collectors.toList());
24500 govind 677
 
35569 amit 678
        Map<Integer, List<Activity>> activityMap = new HashMap<>();
33864 ranu 679
        Map<Integer, List<Activity>> activityMapWithActivityId = new HashMap<>();
35569 amit 680
        Map<Integer, AuthUser> authUserMap = new HashMap<>();
24500 govind 681
 
31762 tejbeer 682
        if (!ticketIds.isEmpty()) {
35569 amit 683
            List<Activity> allActivities = activityRepository.selectAll(ticketIds);
684
            activityMap = allActivities.stream().collect(Collectors.groupingBy(Activity::getTicketId));
685
            activityMapWithActivityId = allActivities.stream().collect(Collectors.groupingBy(Activity::getId));
24787 govind 686
 
35569 amit 687
            Set<Integer> activityCreatorIds = allActivities.stream()
688
                    .map(Activity::getCreatedBy)
689
                    .filter(id -> id > 0)
690
                    .collect(Collectors.toSet());
691
            if (!activityCreatorIds.isEmpty()) {
692
                authUserMap = authRepository.selectByIds(new ArrayList<>(activityCreatorIds))
693
                        .stream().collect(Collectors.toMap(AuthUser::getId, x -> x));
694
            }
31762 tejbeer 695
        }
24787 govind 696
 
35626 amit 697
        int currentUserId;
698
        UserType currentUserType;
699
        if (isAdmin) {
700
            currentUserId = currentAuthUser.getId();
701
            currentUserType = UserType.AUTH_USER;
702
        } else {
703
            currentUserId = loginDetails.getFofoId();
704
            currentUserType = UserType.PARTNER;
705
        }
706
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, currentUserId, currentUserType);
707
        Map<Integer, Activity> lastActivityMap = csService.getLastActivitiesForTickets(ticketIds);
35640 amit 708
        Map<Integer, Activity> lastMessageMap = csService.getLastMessageActivitiesForTickets(ticketIds);
35626 amit 709
        model.addAttribute("unreadMap", unreadMap);
710
        model.addAttribute("lastActivityMap", lastActivityMap);
35640 amit 711
        model.addAttribute("lastMessageMap", lastMessageMap);
35626 amit 712
 
31762 tejbeer 713
        model.addAttribute("tickets", tickets);
714
        model.addAttribute("resolved", ActivityType.RESOLVED);
715
        model.addAttribute("resolved-accepted", ActivityType.RESOLVED_ACCEPTED);
716
        model.addAttribute("resolved-rejected", ActivityType.RESOLVED_REJECTED);
717
        model.addAttribute("authUserIdAndAuthUserMap", authUserIdAndAuthUserMap);
36040 ranu 718
        model.addAttribute("authUserListMap", authUserListMap);
31762 tejbeer 719
        model.addAttribute("subCategoryIdAndSubCategoryMap", subCategoryIdAndSubCategoryMap);
27318 amit.gupta 720
 
31762 tejbeer 721
        model.addAttribute("subCategoryIdAndCategoryMap", subCategoryIdAndCategoryMap);
722
        model.addAttribute("activityMap", activityMap);
33864 ranu 723
        model.addAttribute("authUserMap", authUserMap);
724
        model.addAttribute("activityMapWithActivityId", activityMapWithActivityId);
24500 govind 725
 
31762 tejbeer 726
        model.addAttribute("ticketStatusValues", TicketStatus.values());
727
        model.addAttribute("orderByValues", SortOrder.values());
728
        model.addAttribute("selectedticketStatus", ticketStatus);
729
        model.addAttribute("selectedorderby", sortOrder);
730
        model.addAttribute("ticketSearchTypes", TicketSearchType.values());
731
        model.addAttribute("ticketSearchType", ticketSearchType);
732
        model.addAttribute("searchTerm", searchTerm);
733
    }
24500 govind 734
 
27124 amit.gupta 735
 
31762 tejbeer 736
    @GetMapping(value = "/cs/getActivities")
737
    public String getActivity(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, Model model) throws Exception {
35592 amit 738
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
739
        Ticket ticket = ticketRepository.selectById(ticketId);
740
 
741
        if (ticket == null) {
742
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
743
        }
744
 
745
        // Authorization check: verify user has access to this ticket
746
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
747
        if (!isAdmin) {
748
            // Partners can only view their own tickets
749
            if (ticket.getFofoId() != loginDetails.getFofoId()) {
750
                throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to view this ticket");
751
            }
752
        } else {
753
            // Admins must be assigned to the ticket OR be CRM user OR be in the escalation chain
754
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
755
            boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
756
 
757
            if (!isCrmUser) {
758
                // Check if user is assigned or in escalation chain
759
                List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
760
                boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
761
                boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
762
                        ticket.getL2AuthUser() == currentAuthUser.getId() ||
763
                        ticket.getL3AuthUser() == currentAuthUser.getId() ||
764
                        ticket.getL4AuthUser() == currentAuthUser.getId() ||
765
                        ticket.getL5AuthUser() == currentAuthUser.getId();
766
 
36231 ranu 767
                // Check if user is related to this partner (RBM, ASM, BM etc.)
768
                boolean isRelatedToPartner = false;
35592 amit 769
                if (!isAssigned && !isInEscalationChain) {
36231 ranu 770
                    List<AuthUser> relatedUsers = csService.getAuthUserIdByPartnerId(ticket.getFofoId());
771
                    isRelatedToPartner = relatedUsers.stream()
772
                            .anyMatch(au -> au.getId() == currentAuthUser.getId());
773
                    if (!isRelatedToPartner) {
774
                        throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to view this ticket");
775
                    }
35592 amit 776
                }
36231 ranu 777
 
778
                // User can view but cannot take action if only related to partner (not assigned/escalation)
779
                model.addAttribute("canTakeAction", isAssigned || isInEscalationChain);
35592 amit 780
            }
781
        }
782
 
31762 tejbeer 783
        List<Activity> allactivities = activityRepository.selectAll(ticketId);
35570 amit 784
 
785
        // Batch fetch all documents for attachments (fix N+1 query)
786
        List<Integer> documentIds = allactivities.stream()
787
                .flatMap(a -> a.getActivityAttachment().stream())
788
                .map(ActivityAttachment::getDocumentId)
789
                .distinct()
790
                .collect(Collectors.toList());
791
 
792
        Map<Integer, Document> documentMap = Collections.emptyMap();
793
        if (!documentIds.isEmpty()) {
794
            documentMap = documentRepository.selectByIds(documentIds).stream()
795
                    .collect(Collectors.toMap(Document::getId, d -> d));
796
        }
797
 
798
        // Set document names transiently (do not persist during GET)
799
        for (Activity activity : allactivities) {
800
            for (ActivityAttachment attachment : activity.getActivityAttachment()) {
801
                Document document = documentMap.get(attachment.getDocumentId());
802
                if (document != null) {
803
                    attachment.setDocumentName(document.getDisplayName());
804
                }
31762 tejbeer 805
            }
806
        }
807
        List<Activity> activities = null;
35592 amit 808
        if (isAdmin) {
31762 tejbeer 809
            Set<Integer> authUserIds = allactivities.stream().map(x -> x.getCreatedBy()).collect(Collectors.toSet());
36040 ranu 810
 
811
            // Fetch current assignees for this ticket to show in ASSIGNED/ESCALATED activities
812
            List<TicketAssigned> ticketAssignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
813
            Set<Integer> assigneeIds = ticketAssignments.stream()
814
                    .map(TicketAssigned::getAssineeId)
815
                    .collect(Collectors.toSet());
816
            authUserIds.addAll(assigneeIds);
817
 
35395 amit 818
            List<AuthUser> users = authRepository.selectByIds(new ArrayList<>(authUserIds));
31762 tejbeer 819
            Map<Integer, String> authUserNameMap = users.stream().collect(Collectors.toMap(AuthUser::getId, x -> x.getFirstName() + " " + x.getLastName()));
36040 ranu 820
 
821
            // Get current assignee name (most recent assignment)
822
            String currentAssigneeName = null;
823
            if (!ticketAssignments.isEmpty()) {
824
                int latestAssigneeId = ticketAssignments.get(ticketAssignments.size() - 1).getAssineeId();
825
                currentAssigneeName = authUserNameMap.get(latestAssigneeId);
826
            }
827
            final String assigneeName = currentAssigneeName;
828
 
829
            allactivities.stream().forEach(x -> {
830
                x.setName(authUserNameMap.get(x.getCreatedBy()));
831
                // For ASSIGNED/ESCALATED activities, use name field to show assignee
832
                ActivityType actType = x.getType();
833
                if (assigneeName != null && (actType == ActivityType.ASSIGNED || actType == ActivityType.ESCALATED || actType == ActivityType.ESCALATED_FINAL)) {
834
                    x.setName(assigneeName);
835
                }
836
            });
31762 tejbeer 837
            activities = allactivities;
838
        } else {
839
            activities = allactivities.stream().filter(x -> ActivityType.PARTNER_ACTIVITIES.contains(x.getType())).collect(Collectors.toList());
840
        }
841
        if (activities == null) {
842
            throw new ProfitMandiBusinessException("Activity", ticketId, "No Activity Found");
843
        }
844
        model.addAttribute("response1", mvcResponseSender.createResponseString(activities));
845
        return "response";
24500 govind 846
 
31762 tejbeer 847
    }
24620 govind 848
 
31762 tejbeer 849
    @PostMapping(value = "/cs/createActivity")
850
    public String createActivity(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "assigneeId", defaultValue = "0") int assigneeId, @RequestParam(name = "internal", defaultValue = "true") boolean internal, @RequestParam(name = "message", defaultValue = "") String message, @RequestBody List<Integer> documentIds,
24620 govind 851
 
31762 tejbeer 852
                                 Model model) throws Exception {
24620 govind 853
 
31762 tejbeer 854
        LOGGER.info("documentIds" + documentIds);
855
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
856
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 857
 
858
        if (ticket == null) {
859
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
860
        }
861
 
862
        // Authorization check: verify user has access to add activity to this ticket
863
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
864
        if (!isAdmin) {
865
            // Partners can only add activity to their own tickets
866
            if (ticket.getFofoId() != loginDetails.getFofoId()) {
867
                throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to add activity to this ticket");
868
            }
869
        } else {
870
            // Admins must be assigned to the ticket OR be CRM user OR be in the escalation chain
871
            AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
872
            boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
873
 
874
            if (!isCrmUser) {
875
                List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
876
                boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
877
                boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
878
                        ticket.getL2AuthUser() == currentAuthUser.getId() ||
879
                        ticket.getL3AuthUser() == currentAuthUser.getId() ||
880
                        ticket.getL4AuthUser() == currentAuthUser.getId() ||
881
                        ticket.getL5AuthUser() == currentAuthUser.getId();
882
 
883
                if (!isAssigned && !isInEscalationChain) {
884
                    throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to add activity to this ticket");
885
                }
886
            }
887
        }
31762 tejbeer 888
        List<TicketAssigned> ticketAssignedList = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
889
        List<Integer> authUserIds = ticketAssignedList.stream().map(x -> x.getAssineeId()).collect(Collectors.toList());
890
        authUserIds.add(ticketAssignedList.get(ticketAssignedList.size() - 1).getManagerId());
35395 amit 891
        Map<Integer, AuthUser> authUsersMap = authRepository.selectByIds(authUserIds).stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
31762 tejbeer 892
        if (ticket.getCloseTimestamp() == null) {
893
            Activity activity = new Activity();
894
            activity.setCreatedBy(0);
895
            activity.setCreateTimestamp(LocalDateTime.now());
896
            String subject = null;
897
            String mailMessage = null;
898
            activity.setMessage(message);
899
            if (!roleManager.isAdmin(new HashSet<>(loginDetails.getRoleIds()))) {
900
                CustomRetailer customRetailer = retailerService.getFofoRetailers(true).get(loginDetails.getFofoId());
901
                activity.setType(ActivityType.COMMUNICATION_IN);
902
                subject = String.format("Ticket Update #%s by franchisee %s", ticket.getId(), customRetailer.getBusinessName() + "(" + customRetailer.getCode() + ")");
903
                mailMessage = String.format("Franchisee message - %s", message);
904
            } else {
905
                AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
906
                activity.setCreatedBy(authUser.getId());
907
                authUsersMap.remove(authUser.getId());
908
                subject = String.format("Ticket Update #%s by %s", ticket.getId(), authUser.getName());
909
                mailMessage = String.format("%s's message - %s", authUser.getFirstName(), message);
35569 amit 910
                // Only CRM users can send external communications
911
                boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
912
                if (internal || !isCrmUser) {
31762 tejbeer 913
                    activity.setType(ActivityType.COMMUNICATION_INTERNAL);
914
                } else {
915
                    String updatedBy = "SD Team";
31854 amit.gupta 916
                    CustomRetailer customRetailer = retailerService.getFofoRetailers(false).get(ticket.getFofoId());
31762 tejbeer 917
                    subject = String.format("Ticket Update #%s by %s", ticket.getId(), updatedBy);
918
                    String partnerMessage = String.format("%s's message - %s", updatedBy, message);
919
                    this.activityRelatedMail(customRetailer.getEmail(), null, "subject", partnerMessage);
920
                    activity.setType(ActivityType.COMMUNICATION_OUT);
921
                }
922
            }
923
            activityRepository.persist(activity);
924
 
925
            for (Integer documentId : documentIds) {
926
                ActivityAttachment activityAttachment = new ActivityAttachment();
927
                activityAttachment.setActivityId(activity.getId());
928
                activityAttachment.setDocumentId(documentId);
929
                activityAttachmentRepository.persist(activityAttachment);
930
            }
931
 
932
            csService.addActivity(ticket, activity);
933
            AuthUser authUser = authUsersMap.remove(authUserIds.get(0));
934
            if (authUser == null) {
935
                authUser = authUsersMap.remove(authUserIds.get(1));
936
            }
937
            model.addAttribute("response1", mvcResponseSender.createResponseString(authUser));
938
            String[] cc = authUsersMap.entrySet().stream().map(x -> x.getValue().getEmailId()).toArray(String[]::new);
939
            this.activityRelatedMail(authUser.getEmailId(), cc, subject, mailMessage);
940
        } else {
941
            throw new ProfitMandiBusinessException("Ticket", ticket.getId(), "Already closed ticket");
942
        }
943
        return "response";
944
    }
945
 
946
    private void activityRelatedMail(String to, String[] cc, String subject, String message) throws ProfitMandiBusinessException {
947
        try {
35957 amit 948
            mailOutboxService.queueMail(to, cc, subject, message, "CsController.activityRelatedMail");
31762 tejbeer 949
        } catch (Exception e) {
950
            throw new ProfitMandiBusinessException("Ticket Activity", to, "Could not send ticket activity mail");
951
        }
952
    }
953
 
954
    @PostMapping(value = "/cs/closeTicket")
955
    public String closeTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "happyCode") String happyCode, Model model) throws Exception {
35592 amit 956
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
31762 tejbeer 957
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 958
 
959
        if (ticket == null) {
960
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
961
        }
962
 
963
        // Authorization check: only the ticket owner (partner) can close the ticket
964
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
965
        if (isAdmin) {
966
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Only partners can close tickets using happy code");
967
        }
968
 
969
        // Verify the partner owns this ticket
970
        if (ticket.getFofoId() != loginDetails.getFofoId()) {
971
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to close this ticket");
972
        }
973
 
31762 tejbeer 974
        if (ticket.getHappyCode().equals(happyCode)) {
975
            ticket.setCloseTimestamp(LocalDateTime.now());
976
            ticketRepository.persist(ticket);
977
            model.addAttribute("response1", mvcResponseSender.createResponseString(true));
978
        } else {
979
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Happy Code can't match");
980
        }
981
        return "response";
982
    }
983
 
34913 ranu 984
    @GetMapping(value = "/cs/myPartyTicketTicket")
985
    public String getMyPartyTicketTicket(HttpServletRequest request, @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder, @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus, @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType, @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm, Model model) throws ProfitMandiBusinessException {
35626 amit 986
        populateMyPartnerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, model);
987
        return "my-partner-tickets";
988
    }
989
 
990
    @GetMapping(value = "/cs/myPartyTicket-content")
991
    public String getMyPartyTicketContent(HttpServletRequest request,
992
                                          @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
993
                                          @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
994
                                          @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
995
                                          @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
996
                                          Model model) throws ProfitMandiBusinessException {
997
        populateMyPartnerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, model);
998
        return "my-partner-tickets-content";
999
    }
1000
 
1001
    private void populateMyPartnerTicketModel(HttpServletRequest request, SortOrder sortOrder, TicketStatus ticketStatus,
1002
                                              TicketSearchType ticketSearchType, int searchTerm, Model model) throws ProfitMandiBusinessException {
34913 ranu 1003
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1004
        AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1005
        List<Ticket> tickets = new ArrayList<>();
1006
        Map<String, Set<Integer>> storeGuyMap = csService.getAuthUserPartnerIdMapping();
1007
        Set<Integer> fofoIds = storeGuyMap.get(authUser.getEmailId());
1008
 
1009
        Map<Integer, List<AuthUser>> authUserListMap = null;
1010
        if (fofoIds != null && !fofoIds.isEmpty()) {
35571 amit 1011
            List<Ticket> allPartnerTickets = ticketRepository.selectAllOpenTickets(new ArrayList<>(fofoIds));
1012
            if (allPartnerTickets != null && !allPartnerTickets.isEmpty()) {
1013
                tickets = allPartnerTickets.stream()
1014
                        .filter(ticket -> ticket.getLastActivity() == null ||
1015
                                ticket.getLastActivity() != ActivityType.RESOLVED)
1016
                        .collect(Collectors.toList());
34913 ranu 1017
            }
1018
        }
1019
 
1020
        authUserListMap = csService.getAssignedAuthList(tickets);
1021
 
36231 ranu 1022
        // Find which tickets are assigned to or escalated to this user (can take action)
1023
        Set<Integer> actionableTicketIds = new HashSet<>();
34913 ranu 1024
        if (tickets.size() > 0) {
36231 ranu 1025
            List<TicketAssigned> allAssignments = ticketAssignedRepository.selectByTicketIds(
1026
                    tickets.stream().map(Ticket::getId).collect(Collectors.toList()));
1027
            for (TicketAssigned ta : allAssignments) {
1028
                if (ta.getAssineeId() == authUser.getId() || ta.getManagerId() == authUser.getId()) {
1029
                    actionableTicketIds.add(ta.getTicketId());
1030
                }
1031
            }
1032
            // Also check escalation chain
1033
            for (Ticket t : tickets) {
1034
                if (t.getL1AuthUser() == authUser.getId() || t.getL2AuthUser() == authUser.getId()
1035
                        || t.getL3AuthUser() == authUser.getId() || t.getL4AuthUser() == authUser.getId()
1036
                        || t.getL5AuthUser() == authUser.getId()) {
1037
                    actionableTicketIds.add(t.getId());
1038
                }
1039
            }
34913 ranu 1040
            Map<Integer, CustomRetailer> fofoIdsAndCustomRetailer = csService.getPartnerByFofoIds(tickets);
1041
            model.addAttribute("fofoIdsAndCustomRetailer", fofoIdsAndCustomRetailer);
1042
        }
36231 ranu 1043
        model.addAttribute("actionableTicketIds", actionableTicketIds);
34913 ranu 1044
 
35626 amit 1045
        // Pagination for partner tickets (in-memory since selectAllOpenTickets doesn't support DB-level pagination)
1046
        long totalRecords = tickets.size();
1047
        int totalPages = totalRecords > 0 ? (int) Math.ceil((double) totalRecords / 25) : 1;
1048
        model.addAttribute("totalRecords", totalRecords);
1049
        model.addAttribute("currentPage", 0);
1050
        model.addAttribute("pageSize", 25);
1051
        model.addAttribute("totalPages", totalPages);
1052
        model.addAttribute("startRecord", totalRecords > 0 ? 1 : 0);
1053
        model.addAttribute("endRecord", totalRecords);
1054
        model.addAttribute("currentPageDisplay", 1);
1055
        model.addAttribute("prevPage", 0);
1056
        model.addAttribute("nextPage", 1);
1057
        model.addAttribute("lastPage", totalPages - 1);
1058
 
34913 ranu 1059
        model.addAttribute("size", tickets.size());
1060
        model.addAttribute("tickets", tickets);
1061
 
1062
        List<Integer> subCategoryIds = tickets.stream().map(x -> x.getSubCategoryId()).collect(Collectors.toList());
1063
        Map<Integer, TicketSubCategory> subCategoryIdAndSubCategoryMap = csService.getSubCategoryIdAndSubCategoryMap(subCategoryIds);
35571 amit 1064
        if (subCategoryIdAndSubCategoryMap == null) {
1065
            subCategoryIdAndSubCategoryMap = new HashMap<>();
1066
        }
34913 ranu 1067
 
1068
        Map<Integer, TicketCategory> subCategoryIdAndCategoryMap = csService.getSubCategoryIdAndCategoryMap(subCategoryIds);
35571 amit 1069
        if (subCategoryIdAndCategoryMap == null) {
1070
            subCategoryIdAndCategoryMap = new HashMap<>();
1071
        }
34913 ranu 1072
 
1073
        List<Integer> ticketIds = tickets.stream().map(x -> x.getId()).collect(Collectors.toList());
1074
        Map<Integer, List<Activity>> activityMap = new HashMap<>();
1075
        Map<Integer, List<Activity>> activityMapWithActivityId = new HashMap<>();
35569 amit 1076
        Map<Integer, AuthUser> authUserMap = new HashMap<>();
34913 ranu 1077
 
1078
        if (!ticketIds.isEmpty()) {
35569 amit 1079
            List<Activity> allActivities = activityRepository.selectAll(ticketIds);
1080
            activityMap = allActivities.stream().collect(Collectors.groupingBy(Activity::getTicketId));
1081
            activityMapWithActivityId = allActivities.stream().collect(Collectors.groupingBy(Activity::getId));
34913 ranu 1082
 
35569 amit 1083
            Set<Integer> activityCreatorIds = allActivities.stream()
1084
                    .map(Activity::getCreatedBy)
1085
                    .filter(id -> id > 0)
1086
                    .collect(Collectors.toSet());
1087
            if (!activityCreatorIds.isEmpty()) {
1088
                authUserMap = authRepository.selectByIds(new ArrayList<>(activityCreatorIds))
1089
                        .stream().collect(Collectors.toMap(AuthUser::getId, x -> x));
1090
            }
34913 ranu 1091
        }
1092
 
35626 amit 1093
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, authUser.getId(), UserType.AUTH_USER);
35640 amit 1094
        List<Integer> partnerTicketIds = tickets.stream().map(Ticket::getId).collect(Collectors.toList());
1095
        Map<Integer, Activity> lastActivityMap = csService.getLastActivitiesForTickets(partnerTicketIds);
1096
        Map<Integer, Activity> lastMessageMap = csService.getLastMessageActivitiesForTickets(partnerTicketIds);
35626 amit 1097
        model.addAttribute("unreadMap", unreadMap);
1098
        model.addAttribute("lastActivityMap", lastActivityMap);
35640 amit 1099
        model.addAttribute("lastMessageMap", lastMessageMap);
35626 amit 1100
 
34913 ranu 1101
        model.addAttribute("ticketStatusValues", TicketStatus.values());
1102
        model.addAttribute("orderByValues", SortOrder.values());
1103
        model.addAttribute("selectedticketStatus", ticketStatus);
1104
        model.addAttribute("selectedorderby", sortOrder);
1105
        model.addAttribute("ticketSearchTypes", TicketSearchType.values());
1106
        model.addAttribute("ticketSearchType", ticketSearchType);
1107
        model.addAttribute("searchTerm", searchTerm);
35571 amit 1108
        model.addAttribute("authUserListMap", authUserListMap != null ? authUserListMap : new HashMap<>());
34913 ranu 1109
        model.addAttribute("subCategoryIdAndSubCategoryMap", subCategoryIdAndSubCategoryMap);
1110
 
1111
        model.addAttribute("subCategoryIdAndCategoryMap", subCategoryIdAndCategoryMap);
1112
 
1113
        model.addAttribute("activityMap", activityMap);
1114
        model.addAttribute("authUserMap", authUserMap);
1115
        model.addAttribute("activityMapWithActivityId", activityMapWithActivityId);
35569 amit 1116
        boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1117
        model.addAttribute("isCrmUser", isCrmUser);
35626 amit 1118
    }
34913 ranu 1119
 
35626 amit 1120
    @GetMapping(value = "/cs/managerTicket")
1121
    public String getManagerTickets(HttpServletRequest request,
1122
                                    @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
1123
                                    @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
1124
                                    @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
1125
                                    @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
1126
                                    @RequestParam(name = "page", defaultValue = "0") int page,
1127
                                    @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
1128
                                    @RequestParam(name = "search", required = false) String searchText,
1129
                                    Model model) throws ProfitMandiBusinessException {
1130
        populateManagerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
1131
        return "managerTicket";
34913 ranu 1132
    }
1133
 
35626 amit 1134
    @GetMapping(value = "/cs/managerTicket-content")
1135
    public String getManagerTicketContent(HttpServletRequest request,
1136
                                          @RequestParam(name = "orderby", defaultValue = "DESCENDING") SortOrder sortOrder,
1137
                                          @RequestParam(name = "ticketStatus", defaultValue = "OPENED") TicketStatus ticketStatus,
1138
                                          @RequestParam(name = "ticketSearchType", defaultValue = "") TicketSearchType ticketSearchType,
1139
                                          @RequestParam(name = "searchTerm", defaultValue = "0") int searchTerm,
1140
                                          @RequestParam(name = "page", defaultValue = "0") int page,
1141
                                          @RequestParam(name = "pageSize", defaultValue = "25") int pageSize,
1142
                                          @RequestParam(name = "search", required = false) String searchText,
1143
                                          Model model) throws ProfitMandiBusinessException {
1144
        populateManagerTicketModel(request, sortOrder, ticketStatus, ticketSearchType, searchTerm, page, pageSize, searchText, model);
1145
        return "managerTicket-content";
1146
    }
1147
 
1148
    private void populateManagerTicketModel(HttpServletRequest request, SortOrder sortOrder, TicketStatus ticketStatus,
1149
                                            TicketSearchType ticketSearchType, int searchTerm, int page, int pageSize,
1150
                                            String searchText, Model model) throws ProfitMandiBusinessException {
31762 tejbeer 1151
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
35626 amit 1152
        long totalRecords = 0;
31762 tejbeer 1153
        AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
35626 amit 1154
 
1155
        // Block L1-only users from accessing manager tickets
1156
        List<Position> positions = positionRepository.selectAllByAuthUserId(authUser.getId());
1157
        boolean isAboveL1 = positions.stream().anyMatch(pos -> pos.getEscalationType() != EscalationType.L1);
1158
        if (!isAboveL1) {
1159
            throw new ProfitMandiBusinessException("ManagerTicket", 0, "Access denied: requires at least L2 position");
1160
        }
1161
 
35594 amit 1162
        boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1163
 
35626 amit 1164
        List<Integer> userCategoryIds = positionRepository.selectCategoryIdsByAuthUserId(authUser.getId());
1165
 
31762 tejbeer 1166
        List<Ticket> tickets = null;
1167
        Map<Integer, List<AuthUser>> authUserListMap = null;
35594 amit 1168
 
35626 amit 1169
        int offset = page * pageSize;
1170
        Optional<Boolean> resolvedOpt = ticketStatus.equals(TicketStatus.RESOLVED) ? Optional.empty() : Optional.of(TicketStatus.CLOSED.equals(ticketStatus));
1171
 
35594 amit 1172
        if (isCrmUser) {
35626 amit 1173
            tickets = ticketRepository.selectAllTicketsPaginated(
1174
                    resolvedOpt, sortOrder, ticketSearchType, searchTerm, searchText,
1175
                    userCategoryIds, offset, pageSize);
1176
            totalRecords = ticketRepository.selectAllTicketsPaginatedCount(
1177
                    resolvedOpt, ticketSearchType, searchTerm, searchText, userCategoryIds);
31762 tejbeer 1178
        } else {
35626 amit 1179
            tickets = ticketRepository.selectAllManagerTicketPaginated(
1180
                    authUser.getId(), sortOrder, resolvedOpt, ticketSearchType, searchTerm, searchText,
1181
                    offset, pageSize);
1182
            totalRecords = ticketRepository.selectAllCountByManagerTicketPaginated(
1183
                    authUser.getId(), resolvedOpt, ticketSearchType, searchTerm, searchText);
31762 tejbeer 1184
        }
1185
        authUserListMap = csService.getAssignedAuthList(tickets);
1186
 
1187
        if (tickets.size() > 0) {
1188
            Map<Integer, CustomRetailer> fofoIdsAndCustomRetailer = csService.getPartnerByFofoIds(tickets);
1189
            model.addAttribute("fofoIdsAndCustomRetailer", fofoIdsAndCustomRetailer);
1190
        }
1191
 
35626 amit 1192
        int totalPages = (int) Math.ceil((double) totalRecords / pageSize);
1193
        if (totalPages == 0) totalPages = 1;
1194
        int startRecord = totalRecords > 0 ? offset + 1 : 0;
1195
        int endRecord = (int) Math.min(offset + pageSize, totalRecords);
1196
 
1197
        model.addAttribute("size", totalRecords);
1198
        model.addAttribute("totalRecords", totalRecords);
1199
        model.addAttribute("currentPage", page);
1200
        model.addAttribute("pageSize", pageSize);
1201
        model.addAttribute("totalPages", totalPages);
1202
        model.addAttribute("startRecord", startRecord);
1203
        model.addAttribute("endRecord", endRecord);
1204
        model.addAttribute("searchText", searchText != null ? searchText : "");
1205
        model.addAttribute("currentPageDisplay", page + 1);
1206
        model.addAttribute("prevPage", page - 1);
1207
        model.addAttribute("nextPage", page + 1);
1208
        model.addAttribute("lastPage", totalPages - 1);
31762 tejbeer 1209
        model.addAttribute("tickets", tickets);
1210
 
1211
        List<Integer> subCategoryIds = tickets.stream().map(x -> x.getSubCategoryId()).collect(Collectors.toList());
1212
        Map<Integer, TicketSubCategory> subCategoryIdAndSubCategoryMap = csService.getSubCategoryIdAndSubCategoryMap(subCategoryIds);
35571 amit 1213
        if (subCategoryIdAndSubCategoryMap == null) {
1214
            subCategoryIdAndSubCategoryMap = new HashMap<>();
1215
        }
31762 tejbeer 1216
 
1217
        Map<Integer, TicketCategory> subCategoryIdAndCategoryMap = csService.getSubCategoryIdAndCategoryMap(subCategoryIds);
35571 amit 1218
        if (subCategoryIdAndCategoryMap == null) {
1219
            subCategoryIdAndCategoryMap = new HashMap<>();
1220
        }
31762 tejbeer 1221
 
1222
        List<Integer> ticketIds = tickets.stream().map(x -> x.getId()).collect(Collectors.toList());
1223
        Map<Integer, List<Activity>> activityMap = new HashMap<>();
33778 ranu 1224
        Map<Integer, List<Activity>> activityMapWithActivityId = new HashMap<>();
35570 amit 1225
        Map<Integer, AuthUser> authUserMap = new HashMap<>();
31762 tejbeer 1226
 
1227
        if (!ticketIds.isEmpty()) {
35570 amit 1228
            List<Activity> allActivities = activityRepository.selectAll(ticketIds);
1229
            activityMap = allActivities.stream().collect(Collectors.groupingBy(Activity::getTicketId));
1230
            activityMapWithActivityId = allActivities.stream().collect(Collectors.groupingBy(Activity::getId));
31762 tejbeer 1231
 
35570 amit 1232
            Set<Integer> activityCreatorIds = allActivities.stream()
1233
                    .map(Activity::getCreatedBy)
1234
                    .filter(id -> id > 0)
1235
                    .collect(Collectors.toSet());
1236
            if (!activityCreatorIds.isEmpty()) {
1237
                authUserMap = authRepository.selectByIds(new ArrayList<>(activityCreatorIds))
1238
                        .stream().collect(Collectors.toMap(AuthUser::getId, x -> x));
1239
            }
31762 tejbeer 1240
        }
33778 ranu 1241
 
35626 amit 1242
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, authUser.getId(), UserType.AUTH_USER);
35640 amit 1243
        List<Integer> managerTicketIds = tickets.stream().map(Ticket::getId).collect(Collectors.toList());
1244
        Map<Integer, Activity> lastActivityMap = csService.getLastActivitiesForTickets(managerTicketIds);
1245
        Map<Integer, Activity> lastMessageMap = csService.getLastMessageActivitiesForTickets(managerTicketIds);
35626 amit 1246
        model.addAttribute("unreadMap", unreadMap);
1247
        model.addAttribute("lastActivityMap", lastActivityMap);
35640 amit 1248
        model.addAttribute("lastMessageMap", lastMessageMap);
35626 amit 1249
 
31762 tejbeer 1250
        model.addAttribute("ticketStatusValues", TicketStatus.values());
1251
        model.addAttribute("orderByValues", SortOrder.values());
1252
        model.addAttribute("selectedticketStatus", ticketStatus);
1253
        model.addAttribute("selectedorderby", sortOrder);
1254
        model.addAttribute("ticketSearchTypes", TicketSearchType.values());
1255
        model.addAttribute("ticketSearchType", ticketSearchType);
1256
        model.addAttribute("searchTerm", searchTerm);
1257
        model.addAttribute("authUserListMap", authUserListMap);
1258
        model.addAttribute("subCategoryIdAndSubCategoryMap", subCategoryIdAndSubCategoryMap);
1259
 
1260
        model.addAttribute("subCategoryIdAndCategoryMap", subCategoryIdAndCategoryMap);
1261
 
1262
        model.addAttribute("activityMap", activityMap);
33778 ranu 1263
        model.addAttribute("authUserMap", authUserMap);
1264
        model.addAttribute("activityMapWithActivityId", activityMapWithActivityId);
35594 amit 1265
        model.addAttribute("isCrmUser", isCrmUser);
31762 tejbeer 1266
    }
1267
 
1268
 
1269
    @GetMapping(value = "/cs/edit-ticket")
35594 amit 1270
    public String getEditTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, Model model) throws ProfitMandiBusinessException {
1271
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1272
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1273
        boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1274
 
31762 tejbeer 1275
        Ticket ticket = ticketRepository.selectById(ticketId);
35626 amit 1276
        if (ticket == null) {
1277
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1278
        }
1279
 
1280
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1281
        int ticketCategoryId = ticketSubCategory.getCategoryId();
1282
        boolean isVisibleSubCategory = ticketSubCategory.isVisibility();
1283
 
1284
        // Check permissions
1285
        boolean hasPositionInCategory = positionRepository.hasCategory(currentAuthUser.getId(), ticketCategoryId);
1286
        List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
1287
        boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
1288
        boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
1289
                ticket.getL2AuthUser() == currentAuthUser.getId() ||
1290
                ticket.getL3AuthUser() == currentAuthUser.getId() ||
1291
                ticket.getL4AuthUser() == currentAuthUser.getId() ||
1292
                ticket.getL5AuthUser() == currentAuthUser.getId();
1293
 
1294
        boolean canEdit = (isCrmUser && isVisibleSubCategory)
1295
                || hasPositionInCategory
1296
                || isAssigned
1297
                || isInEscalationChain;
1298
 
1299
        if (!canEdit) {
1300
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to edit this ticket");
1301
        }
1302
 
31762 tejbeer 1303
        List<TicketCategory> ticketCategories = csService.getAllTicketCategotyFromSubCategory();
1304
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAll(ticketSubCategory.getCategoryId());
1305
        List<AuthUser> authUsers = authRepository.selectAllActiveUser();
35594 amit 1306
 
35617 amit 1307
        // CRM users should not see invisible subcategories
35594 amit 1308
        if (isCrmUser) {
1309
            ticketSubCategories = ticketSubCategories.stream()
35617 amit 1310
                    .filter(TicketSubCategory::isVisibility)
35594 amit 1311
                    .collect(Collectors.toList());
1312
        }
1313
 
1314
        // Get partner name for modal title
1315
        String partnerName = retailerService.getFofoRetailer(ticket.getFofoId()).getBusinessName();
1316
 
31762 tejbeer 1317
        model.addAttribute("ticket", ticket);
1318
        model.addAttribute("ticketCategories", ticketCategories);
1319
        model.addAttribute("ticketSubCategories", ticketSubCategories);
1320
        model.addAttribute("ticketSubCategory", ticketSubCategory);
1321
        model.addAttribute("authUsers", authUsers);
35594 amit 1322
        model.addAttribute("isCrmUser", isCrmUser);
1323
        model.addAttribute("partnerName", partnerName);
31762 tejbeer 1324
        return "edit-ticket-modal";
1325
    }
1326
 
34913 ranu 1327
    @GetMapping(value = "/cs/edit-partner-ticket")
35594 amit 1328
    public String getEditPartnerTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, Model model) throws ProfitMandiBusinessException {
1329
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1330
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1331
        boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1332
 
34913 ranu 1333
        Ticket ticket = ticketRepository.selectById(ticketId);
1334
        List<TicketCategory> ticketCategories = csService.getAllTicketCategotyFromSubCategory();
1335
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1336
        List<TicketSubCategory> ticketSubCategories = ticketSubCategoryRepository.selectAll(ticketSubCategory.getCategoryId());
1337
        List<AuthUser> authUsers = authRepository.selectAllActiveUser();
35594 amit 1338
 
35617 amit 1339
        // CRM users should not see invisible subcategories
35594 amit 1340
        if (isCrmUser) {
1341
            ticketSubCategories = ticketSubCategories.stream()
35617 amit 1342
                    .filter(TicketSubCategory::isVisibility)
35594 amit 1343
                    .collect(Collectors.toList());
1344
        }
1345
 
1346
        // Get partner name for modal title
1347
        String partnerName = retailerService.getFofoRetailer(ticket.getFofoId()).getBusinessName();
1348
 
34913 ranu 1349
        model.addAttribute("ticket", ticket);
1350
        model.addAttribute("ticketCategories", ticketCategories);
1351
        model.addAttribute("ticketSubCategories", ticketSubCategories);
1352
        model.addAttribute("ticketSubCategory", ticketSubCategory);
1353
        model.addAttribute("authUsers", authUsers);
35594 amit 1354
        model.addAttribute("isCrmUser", isCrmUser);
1355
        model.addAttribute("partnerName", partnerName);
34913 ranu 1356
        return "edit-ticket-partner-modal";
1357
    }
1358
 
31762 tejbeer 1359
    @PostMapping(value = "/cs/edit-ticket")
1360
    public String editTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "subCategoryId", defaultValue = "0") int subCategoryId, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "authUserId", defaultValue = "0") int authUserId, @RequestParam(name = "escalationType", defaultValue = "L1") EscalationType escalationType, Model model) throws Exception {
1361
        LOGGER.info("Ticket Id {}, CategoryId {}, SubCategory Id {} authUserId {}", ticketId, categoryId, subCategoryId, authUserId);
35592 amit 1362
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1363
 
1364
        // Only admins can edit tickets
1365
        if (!roleManager.isAdmin(loginDetails.getRoleIds())) {
1366
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Only admins can edit tickets");
1367
        }
1368
 
31762 tejbeer 1369
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 1370
        if (ticket == null) {
1371
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1372
        }
1373
 
35626 amit 1374
        // Verify admin has access to this ticket
35592 amit 1375
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1376
        boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1377
 
35626 amit 1378
        // Get ticket's subcategory to check visibility
1379
        TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1380
        int ticketCategoryId = ticketSubCategory.getCategoryId();
1381
        boolean isVisibleSubCategory = ticketSubCategory.isVisibility();
35592 amit 1382
 
35626 amit 1383
        // Check if user has position in ticket's category
1384
        boolean hasPositionInCategory = positionRepository.hasCategory(currentAuthUser.getId(), ticketCategoryId);
1385
 
1386
        // Check if assigned or in escalation chain
1387
        List<TicketAssigned> assignments = ticketAssignedRepository.selectByTicketIds(Arrays.asList(ticketId));
1388
        boolean isAssigned = assignments.stream().anyMatch(ta -> ta.getAssineeId() == currentAuthUser.getId() || ta.getManagerId() == currentAuthUser.getId());
1389
        boolean isInEscalationChain = ticket.getL1AuthUser() == currentAuthUser.getId() ||
1390
                ticket.getL2AuthUser() == currentAuthUser.getId() ||
1391
                ticket.getL3AuthUser() == currentAuthUser.getId() ||
1392
                ticket.getL4AuthUser() == currentAuthUser.getId() ||
1393
                ticket.getL5AuthUser() == currentAuthUser.getId();
1394
 
1395
        // Permission rules:
1396
        // 1. CRM user AND ticket has visible subcategory, OR
1397
        // 2. Has position in ticket's category (can edit even invisible), OR
1398
        // 3. Is assigned to ticket, OR
1399
        // 4. Is in escalation chain
1400
        boolean canEdit = (isCrmUser && isVisibleSubCategory)
1401
                || hasPositionInCategory
1402
                || isAssigned
1403
                || isInEscalationChain;
1404
 
1405
        if (!canEdit) {
1406
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to edit this ticket");
35592 amit 1407
        }
1408
 
31762 tejbeer 1409
        csService.updateTicket(categoryId, subCategoryId, ticket, authUserId, escalationType);
1410
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1411
        return "response";
1412
 
1413
    }
1414
 
34913 ranu 1415
    @PostMapping(value = "/cs/edit-partner-ticket")
1416
    public String editPartnerTicket(HttpServletRequest request, @RequestParam(name = "ticketId", defaultValue = "0") int ticketId, @RequestParam(name = "subCategoryId", defaultValue = "0") int subCategoryId, @RequestParam(name = "categoryId", defaultValue = "0") int categoryId, @RequestParam(name = "authUserId", defaultValue = "0") int authUserId, @RequestParam(name = "escalationType", defaultValue = "L1") EscalationType escalationType, Model model) throws Exception {
1417
        LOGGER.info("Ticket Id {}, CategoryId {}, SubCategory Id {} authUserId {}", ticketId, categoryId, subCategoryId, authUserId);
35592 amit 1418
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1419
 
1420
        // Only admins can edit partner tickets
1421
        if (!roleManager.isAdmin(loginDetails.getRoleIds())) {
1422
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Only admins can edit partner tickets");
1423
        }
1424
 
34913 ranu 1425
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 1426
        if (ticket == null) {
1427
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1428
        }
1429
 
1430
        // Verify admin has access (must be in Sales, ABM, or RBM position for the partner)
1431
        AuthUser currentAuthUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1432
        Map<String, Set<Integer>> authUserPartnerMap = csService.getAuthUserPartnerIdMapping();
1433
        Set<Integer> allowedPartnerIds = authUserPartnerMap.get(currentAuthUser.getEmailId());
1434
 
1435
        if (allowedPartnerIds == null || !allowedPartnerIds.contains(ticket.getFofoId())) {
1436
            // Also allow CRM users
1437
            boolean isCrmUser = positionRepository.hasCategory(currentAuthUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1438
            if (!isCrmUser) {
1439
                throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to edit this partner's ticket");
1440
            }
1441
        }
1442
 
34913 ranu 1443
        csService.updateTicket(categoryId, subCategoryId, ticket, authUserId, escalationType);
1444
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1445
        return "response";
1446
 
1447
    }
1448
 
31762 tejbeer 1449
    @PostMapping(value = "/cs/changeTicketAssignee")
1450
    public String changeTicketAssignee(HttpServletRequest request, @RequestParam(name = "positionId", defaultValue = "0") int positionId, Model model) throws Exception {
1451
        Position position = positionRepository.selectById(positionId);
1452
        if (position.isTicketAssignee()) {
1453
            position.setTicketAssignee(false);
1454
        } else {
1455
            position.setTicketAssignee(true);
1456
        }
1457
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1458
        return "response";
1459
    }
1460
 
1461
 
1462
    @DeleteMapping(value = "/cs/removePosition")
1463
    public String removePosition(HttpServletRequest request, @RequestParam(name = "positionId", defaultValue = "0") int positionId, Model model) throws Exception {
1464
        positionRepository.delete(positionId);
1465
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1466
        return "response";
1467
    }
1468
 
1469
    @PostMapping(value = "/cs/create-last-activity")
1470
    public String createlastActivity(HttpServletRequest request, @RequestParam(name = "ticketId") int ticketId, @RequestParam(name = "lastactivity") ActivityType lastActivity, Model model) throws Exception {
1471
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1472
        Ticket ticket = ticketRepository.selectById(ticketId);
35592 amit 1473
 
1474
        if (ticket == null) {
1475
            throw new ProfitMandiBusinessException("Ticket", ticketId, "Ticket not found");
1476
        }
1477
 
1478
        // Authorization check for partners: can only update their own tickets
1479
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
1480
        if (!isAdmin && ticket.getFofoId() != loginDetails.getFofoId()) {
1481
            throw new ProfitMandiBusinessException("Ticket", ticketId, "You do not have permission to update this ticket");
1482
        }
1483
 
31762 tejbeer 1484
        Activity activity = new Activity();
1485
        String subject = String.format(ACTIVITY_SUBJECT, ticket.getId());
35592 amit 1486
        if (isAdmin) {
35603 amit 1487
            // CRM team members can resolve any ticket
1488
            // Sales team members can resolve their own Sales category tickets
1489
            // RBM team members can resolve their own RBM category tickets
35592 amit 1490
            AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1491
            boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
35603 amit 1492
            boolean isSalesUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_SALES);
1493
            boolean isRbmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_RBM);
1494
 
1495
            // Check if ticket belongs to Sales or RBM category
1496
            TicketSubCategory ticketSubCategory = ticketSubCategoryRepository.selectById(ticket.getSubCategoryId());
1497
            boolean isSalesTicket = ticketSubCategory != null && ticketSubCategory.getCategoryId() == ProfitMandiConstants.TICKET_CATEGORY_SALES;
1498
            boolean isRbmTicket = ticketSubCategory != null && ticketSubCategory.getCategoryId() == ProfitMandiConstants.TICKET_CATEGORY_RBM;
1499
 
1500
            // Allow if CRM user OR (Sales user AND Sales ticket) OR (RBM user AND RBM ticket)
1501
            if (!isCrmUser && !(isSalesUser && isSalesTicket) && !(isRbmUser && isRbmTicket)) {
1502
                throw new ProfitMandiBusinessException("Ticket", ticketId, "Only CRM, Sales (for Sales tickets), or RBM (for RBM tickets) team can mark tickets as resolved");
35592 amit 1503
            }
31762 tejbeer 1504
            ticket.setLastActivity(lastActivity);
1505
            String to = retailerService.getFofoRetailer(ticket.getFofoId()).getEmail();
1506
            String message = String.format(PARTNER_RESOLVED_TICKET_MAIL, ticketId, "REOPEN");
1507
            activity.setMessage(message);
35592 amit 1508
            activity.setCreatedBy(authUser.getId());
31762 tejbeer 1509
            activity.setTicketId(ticketId);
1510
            activity.setCreateTimestamp(LocalDateTime.now());
1511
            activity.setType(ActivityType.COMMUNICATION_OUT);
1512
            this.activityRelatedMail(to, null, subject, message);
1513
        } else {
1514
            if (ActivityType.RESOLVED_ACCEPTED == lastActivity) {
1515
                ticket.setLastActivity(lastActivity);
1516
                ticket.setCloseTimestamp(LocalDateTime.now());
1517
                activity.setMessage(ActivityType.RESOLVED_ACCEPTED.toString());
1518
                activity.setCreatedBy(0);
1519
                activity.setTicketId(ticketId);
1520
                activity.setType(ActivityType.COMMUNICATION_IN);
1521
                activity.setCreateTimestamp(LocalDateTime.now());
1522
            } else {
1523
                String message = String.format(INTERNAL_REOPEN_MAIL, ticketId, retailerService.getFofoRetailer(loginDetails.getFofoId()).getBusinessName());
1524
                String to = authRepository.selectById(ticket.getL1AuthUser()).getEmailId();
35395 amit 1525
                String[] ccTo = authRepository.selectByIds(Arrays.asList(ticket.getL2AuthUser(), ticket.getL3AuthUser(), ticket.getL4AuthUser(), ticket.getL5AuthUser())).stream().map(x -> x.getEmailId()).toArray(String[]::new);
31762 tejbeer 1526
                ticket.setLastActivity(lastActivity);
1527
                ticket.setUpdateTimestamp(LocalDateTime.now());
1528
                ticketAssignedRepository.deleteByTicketId(ticketId);
1529
                TicketAssigned ticketAssigned = new TicketAssigned();
1530
                ticketAssigned.setAssineeId(ticket.getL1AuthUser());
1531
                ticketAssigned.setTicketId(ticketId);
1532
                ticketAssignedRepository.persist(ticketAssigned);
1533
                activity.setMessage(INTERNAL_REOPEN_ACTIVITY_MESSAGE);
1534
                activity.setCreatedBy(0);
1535
                activity.setTicketId(ticketId);
1536
                activity.setType(ActivityType.COMMUNICATION_IN);
1537
                activity.setCreateTimestamp(LocalDateTime.now());
1538
                this.activityRelatedMail(to, ccTo, subject, message);
1539
                this.activityRelatedMail(retailerService.getFofoRetailer(loginDetails.getFofoId()).getEmail(), null, subject, String.format(PARTNER_REOPEN, ticketId));
1540
            }
1541
 
1542
        }
1543
        activityRepository.persist(activity);
1544
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1545
        return "response";
1546
    }
1547
 
32812 shampa 1548
 
1549
 
1550
 
1551
 
1552
    @PostMapping(value = "/partner-position/update")
1553
    public String positionUpdated(Model model, @RequestBody List<PartnerPositonUpdateModel> partnerPositionUpdateModels)
1554
            throws Exception {
1555
 
32821 shampa 1556
        Map<Integer, List<String>> positionIdsToAddMap = partnerPositionUpdateModels.stream().filter(x->x.getPositionIdTo()!=0).collect(Collectors.groupingBy(x->x.getPositionIdTo(),
32812 shampa 1557
                Collectors.mapping(x->x.getStoreCode(), Collectors.toList())));
1558
 
32821 shampa 1559
        Map<Integer, List<String>> positionIdsToRemoveMap = partnerPositionUpdateModels.stream().filter(x->x.getPositionIdFrom()!=0).collect(Collectors.groupingBy(x->x.getPositionIdFrom(),
32812 shampa 1560
                Collectors.mapping(x->x.getStoreCode(), Collectors.toList())));
1561
 
1562
        List<Integer> positionIdsToUpdate = new ArrayList<>();
1563
        positionIdsToUpdate.addAll(positionIdsToAddMap.keySet());
1564
        positionIdsToUpdate.addAll(positionIdsToRemoveMap.keySet());
1565
 
35395 amit 1566
        Map<Integer, Position> positionsToUpdateMap =  positionRepository.selectByIds(positionIdsToUpdate).stream().collect(Collectors.toMap(x->x.getId(), x->x));
32812 shampa 1567
        List<Integer> invalidPositionIds = positionsToUpdateMap.values().stream().filter(x-> x.getCategoryId()!= ProfitMandiConstants.TICKET_CATEGORY_RBM
34908 ranu 1568
                && x.getCategoryId() != ProfitMandiConstants.TICKET_CATEGORY_SALES && x.getCategoryId() != ProfitMandiConstants.TICKET_CATEGORY_ABM).map(x -> x.getId()).collect(Collectors.toList());
32812 shampa 1569
        if(invalidPositionIds.size() > 0) {
34908 ranu 1570
            String message = "Non RBM/Sales/ABM are not allowed - " + invalidPositionIds;
32812 shampa 1571
            throw new ProfitMandiBusinessException(message, message, message);
1572
        }
1573
 
1574
        for (Map.Entry<Integer, List<String>> positionIdStoreMapEntry : positionIdsToAddMap.entrySet()) {
1575
            int positionId = positionIdStoreMapEntry.getKey();
1576
            Position position = positionsToUpdateMap.get(positionId);
32821 shampa 1577
            LOGGER.info("positionId - {}, Position - {}", positionId, position);
32812 shampa 1578
            List<String> storeCodesToAdd = positionIdStoreMapEntry.getValue();
1579
            List<Integer> retailerIdsToAdd = fofoStoreRepository.selectByStoreCodes(storeCodesToAdd).stream().map(x->x.getId()).collect(Collectors.toList());
1580
            Map<Integer, PartnerPosition> partnerPositionsMapByFofoId  = partnerPositionRepository
32821 shampa 1581
                    .selectByRegionIdAndPostionId(Arrays.asList(position.getRegionId())
1582
                            ,Arrays.asList(positionId)).stream().collect(Collectors.toMap(x->x.getFofoId(),x->x));
32812 shampa 1583
            for (Integer retailerIdToAdd : retailerIdsToAdd) {
1584
                if (!partnerPositionsMapByFofoId.containsKey(retailerIdToAdd)) {
1585
                    PartnerPosition partnerPositionNew = new PartnerPosition();
1586
                    partnerPositionNew.setPositionId(positionId);
1587
                    partnerPositionNew.setFofoId(retailerIdToAdd);
1588
                    partnerPositionNew.setRegionId(position.getRegionId());
32865 amit.gupta 1589
                    partnerPositionRepository.persist(partnerPositionNew);
32812 shampa 1590
                }
1591
            }
1592
        }
1593
 
1594
        for (Map.Entry<Integer, List<String>> positionIdStoreMapEntry : positionIdsToRemoveMap.entrySet()) {
1595
 
1596
            int positionId = positionIdStoreMapEntry.getKey();
1597
            Position position = positionsToUpdateMap.get(positionId);
1598
            List<String> storeCodesToRemove = positionIdStoreMapEntry.getValue();
1599
            List<Integer> retailerIdsToRemove = fofoStoreRepository.selectByStoreCodes(storeCodesToRemove).stream().map(x->x.getId()).collect(Collectors.toList());
1600
            Map<Integer, PartnerPosition> partnerPositionsMapByFofoId  = partnerPositionRepository
1601
                    .selectByRegionIdAndPostionId(Arrays.asList(position.getRegionId()),Arrays.asList(positionId)).stream().collect(Collectors.toMap(x->x.getFofoId(),x->x));
1602
            for (Integer retailerIdToRemove : retailerIdsToRemove) {
1603
                if (partnerPositionsMapByFofoId.containsKey(retailerIdToRemove)) {
1604
                   PartnerPosition partnerPositionToRemove =  partnerPositionsMapByFofoId.get(retailerIdToRemove);
1605
                   partnerPositionRepository.delete(partnerPositionToRemove);
1606
                }
1607
            }
1608
        }
1609
 
1610
 
1611
 
1612
        /*partnerPositionUpdateModels.str
1613
 
1614
        Map<Integer, Position> positionIdMap = positionsToUpdate.stream().collect(Collectors.toMap(x->x.getId(), x->x));
1615
        for (PartnerPositonUpdateModel partnerPositionUpdateModel : partnerPositionUpdateModels) {
1616
            FofoStore fofoStore = fofoStoreRepository.selectByStoreCode(partnerPositionUpdateModel.getStoreCode());
1617
            Position positionFrom = positionIdMap.get(partnerPositionUpdateModel.getPositionIdFrom());
1618
            Position positionTo = positionIdMap.get(partnerPositionUpdateModel.getPositionIdTo());
1619
            if(positionFrom != null) {
1620
                partnerPositionRepository.selectByRegionIdAndPostionId(Arrays.)
1621
              int regionId = positionFrom.getRegionId()
1622
            }
1623
            if(positionTo != null) {
1624
 
1625
            }
1626
        }*/
1627
        model.addAttribute("response1", mvcResponseSender.createResponseString(true));
1628
        return "response";
1629
 
1630
    }
1631
 
35626 amit 1632
    @PostMapping(value = "/cs/markTicketRead")
1633
    @ResponseBody
1634
    public Map<String, Object> markTicketRead(HttpServletRequest request,
1635
                                              @RequestParam(name = "ticketId") int ticketId,
1636
                                              Model model) throws ProfitMandiBusinessException {
1637
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1638
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
1639
        Map<String, Object> response = new HashMap<>();
32812 shampa 1640
 
35626 amit 1641
        if (isAdmin) {
1642
            AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1643
            csService.markTicketAsRead(ticketId, authUser.getId(), UserType.AUTH_USER);
1644
        } else {
1645
            csService.markTicketAsRead(ticketId, loginDetails.getFofoId(), UserType.PARTNER);
1646
        }
1647
 
1648
        response.put("success", true);
1649
        return response;
1650
    }
1651
 
1652
    @GetMapping(value = "/cs/unreadCount")
1653
    @ResponseBody
1654
    public Map<String, Object> getUnreadCount(HttpServletRequest request) throws ProfitMandiBusinessException {
1655
        LoginDetails loginDetails = cookiesProcessor.getCookiesObject(request);
1656
        boolean isAdmin = roleManager.isAdmin(loginDetails.getRoleIds());
1657
        Map<String, Object> response = new HashMap<>();
1658
 
1659
        List<Ticket> tickets;
1660
        int userId;
1661
        UserType userType;
1662
 
1663
        if (isAdmin) {
1664
            AuthUser authUser = authRepository.selectByEmailOrMobile(loginDetails.getEmailId());
1665
            userId = authUser.getId();
1666
            userType = UserType.AUTH_USER;
1667
            boolean isCrmUser = positionRepository.hasCategory(authUser.getId(), ProfitMandiConstants.TICKET_CATEGORY_CRM);
1668
            List<Integer> userCategoryIds = positionRepository.selectCategoryIdsByAuthUserId(authUser.getId());
1669
            if (isCrmUser) {
1670
                tickets = ticketRepository.selectAllTicketsPaginated(
1671
                        Optional.of(false), null, null, 0, null,
1672
                        userCategoryIds, 0, 1000);
1673
            } else {
1674
                tickets = ticketRepository.selectAllByAssigneePaginated(
1675
                        authUser.getId(), Optional.of(false), null, null, 0, null,
1676
                        0, 1000);
1677
            }
1678
        } else {
1679
            userId = loginDetails.getFofoId();
1680
            userType = UserType.PARTNER;
1681
            tickets = ticketRepository.selectAllByCreator(loginDetails.getFofoId(), Optional.of(true), null);
1682
        }
1683
 
1684
        Map<Integer, Boolean> unreadMap = csService.getUnreadStatusForTickets(tickets, userId, userType);
1685
        long unreadCount = unreadMap.values().stream().filter(v -> v).count();
1686
 
1687
        response.put("success", true);
1688
        response.put("unreadCount", unreadCount);
1689
        return response;
1690
    }
1691
 
24417 govind 1692
}