Subversion Repositories SmartDukaan

Rev

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

Rev Author Line No. Line
22653 ashik.ali 1
package com.spice.profitmandi.service.scheme;
2
 
24307 amit.gupta 3
import com.spice.profitmandi.common.enumuration.ItemType;
29927 amit.gupta 4
import com.spice.profitmandi.common.enumuration.MessageType;
22653 ashik.ali 5
import com.spice.profitmandi.common.exception.ProfitMandiBusinessException;
6
import com.spice.profitmandi.common.model.ProfitMandiConstants;
23019 ashik.ali 7
import com.spice.profitmandi.common.model.SchemeModel;
29927 amit.gupta 8
import com.spice.profitmandi.common.model.SendNotificationModel;
29584 manish 9
import com.spice.profitmandi.common.services.ReporticoService;
27395 amit.gupta 10
import com.spice.profitmandi.common.util.FormattingUtils;
22859 ashik.ali 11
import com.spice.profitmandi.common.util.StringUtils;
30121 amit.gupta 12
import com.spice.profitmandi.common.util.Utils;
34221 tejus.loha 13
import com.spice.profitmandi.dao.entity.catalog.*;
29927 amit.gupta 14
import com.spice.profitmandi.dao.entity.fofo.*;
31903 amit.gupta 15
import com.spice.profitmandi.dao.entity.inventory.PartnerAgeingModel;
34709 amit.gupta 16
import com.spice.profitmandi.dao.entity.transaction.Loan;
26498 amit.gupta 17
import com.spice.profitmandi.dao.entity.transaction.PriceDrop;
32060 amit.gupta 18
import com.spice.profitmandi.dao.entity.transaction.UserWallet;
31609 amit.gupta 19
import com.spice.profitmandi.dao.entity.transaction.UserWalletHistory;
23527 ashik.ali 20
import com.spice.profitmandi.dao.enumuration.catalog.AmountType;
22653 ashik.ali 21
import com.spice.profitmandi.dao.enumuration.catalog.SchemeType;
34317 amit.gupta 22
import com.spice.profitmandi.dao.enumuration.catalog.StockTransactionType;
34504 amit.gupta 23
import com.spice.profitmandi.dao.enumuration.fofo.ScanType;
27377 amit.gupta 24
import com.spice.profitmandi.dao.enumuration.transaction.SchemePayoutStatus;
31903 amit.gupta 25
import com.spice.profitmandi.dao.model.AmountModel;
25503 amit.gupta 26
import com.spice.profitmandi.dao.model.CreateSchemeRequest;
31170 amit.gupta 27
import com.spice.profitmandi.dao.repository.catalog.*;
32309 amit.gupta 28
import com.spice.profitmandi.dao.repository.dtr.FofoStoreRepository;
23968 amit.gupta 29
import com.spice.profitmandi.dao.repository.dtr.RetailerRepository;
29927 amit.gupta 30
import com.spice.profitmandi.dao.repository.fofo.*;
34709 amit.gupta 31
import com.spice.profitmandi.dao.repository.transaction.LoanRepository;
26498 amit.gupta 32
import com.spice.profitmandi.dao.repository.transaction.PriceDropRepository;
31609 amit.gupta 33
import com.spice.profitmandi.dao.repository.transaction.UserWalletHistoryRepository;
32060 amit.gupta 34
import com.spice.profitmandi.dao.repository.transaction.UserWalletRepository;
34504 amit.gupta 35
import com.spice.profitmandi.dao.repository.warehouse.AgeingSummaryModel;
33087 amit.gupta 36
import com.spice.profitmandi.dao.repository.warehouse.WarehouseInventoryItemRepository;
29927 amit.gupta 37
import com.spice.profitmandi.service.NotificationService;
23798 amit.gupta 38
import com.spice.profitmandi.service.authentication.RoleManager;
31903 amit.gupta 39
import com.spice.profitmandi.service.inventory.AgeingService;
26722 amit.gupta 40
import com.spice.profitmandi.service.inventory.PurchaseService;
31903 amit.gupta 41
import com.spice.profitmandi.service.pricecircular.CatalogSummaryModel;
31921 amit.gupta 42
import com.spice.profitmandi.service.pricecircular.PriceCircularService;
31903 amit.gupta 43
import com.spice.profitmandi.service.pricecircular.SchemeSummaryModel;
22859 ashik.ali 44
import com.spice.profitmandi.service.wallet.WalletService;
45
import in.shop2020.model.v1.order.WalletReferenceType;
29927 amit.gupta 46
import org.apache.logging.log4j.LogManager;
47
import org.apache.logging.log4j.Logger;
48
import org.hibernate.Session;
49
import org.hibernate.SessionFactory;
50
import org.hibernate.query.Query;
51
import org.springframework.beans.factory.annotation.Autowired;
52
import org.springframework.beans.factory.annotation.Qualifier;
53
import org.springframework.cache.annotation.Cacheable;
54
import org.springframework.stereotype.Component;
22859 ashik.ali 55
 
31170 amit.gupta 56
import javax.persistence.TypedQuery;
29927 amit.gupta 57
import javax.persistence.criteria.CriteriaBuilder;
58
import javax.persistence.criteria.CriteriaQuery;
59
import javax.persistence.criteria.Predicate;
60
import javax.persistence.criteria.Root;
61
import java.text.MessageFormat;
34115 ranu 62
import java.time.LocalDate;
63
import java.time.LocalDateTime;
64
import java.time.LocalTime;
34504 amit.gupta 65
import java.time.YearMonth;
29927 amit.gupta 66
import java.util.*;
67
import java.util.stream.Collectors;
68
 
22653 ashik.ali 69
@Component
70
public class SchemeServiceImpl implements SchemeService {
71
 
31410 amit.gupta 72
    private static final Logger LOGGER = LogManager.getLogger(SchemeServiceImpl.class);
31916 amit.gupta 73
    private static final Set<Integer> tagIds = new HashSet<>(Arrays.asList(4));
34709 amit.gupta 74
    private static final List<SchemeType> FULL_DAYS_CD_SCHEME_TYPES = Arrays.asList(SchemeType.CASH_DISCOUNT, SchemeType.CASH_DISCOUNT1);
75
    private static final List<SchemeType> HALF_DAYS_CD_REJECT_SCHEME_TYPES = Arrays.asList(SchemeType.CASH_DISCOUNT1);
34017 amit.gupta 76
 
34709 amit.gupta 77
 
34317 amit.gupta 78
    private static final List<SchemeType> ACTIVATION_SCHEME_TYPES = Arrays.asList(SchemeType.SPECIAL_SUPPORT, SchemeType.SELLOUT);
35060 amit 79
    private static final List<SchemeType> NOFITY_SCHEME_TYPES = Arrays.asList(SchemeType.SPECIAL_SUPPORT, SchemeType.SELLOUT, SchemeType.SELLIN);
33600 ranu 80
 
31410 amit.gupta 81
    @Autowired
34221 tejus.loha 82
    SchemeBlockedImeiRepository schemeBlockedImeiRepository;
83
    @Autowired
31410 amit.gupta 84
    StateGstRateRepository stateGstRateRepository;
85
    @Autowired
86
    NotificationService notificationService;
87
    @Autowired
88
    @Qualifier("fofoInventoryItemRepository")
89
    private InventoryItemRepository inventoryItemRepository;
90
    @Autowired
91
    private ActivatedImeiRepository activatedImeiRepository;
92
    @Autowired
93
    private PartnerTypeChangeService partnerTypeChangeService;
94
    @Autowired
95
    private PurchaseService purchaseService;
96
    @Autowired
97
    private ScanRecordRepository scanRecordRepository;
98
    @Autowired
99
    private SessionFactory sessionFactory;
100
    @Autowired
101
    private SchemeRepository schemeRepository;
102
    @Autowired
103
    private PriceDropRepository priceDropRepository;
104
    @Autowired
105
    private RoleManager roleManager;
106
    @Autowired
107
    private RetailerRepository retailerRepository;
108
    @Autowired
109
    private ReporticoService reporticoService;
110
    @Autowired
111
    private TagListingRepository tagListingRepository;
112
    @Autowired
113
    private SchemeInOutRepository schemeInOutRepository;
114
    @Autowired
115
    @Qualifier("catalogItemRepository")
116
    private ItemRepository itemRepository;
117
    @Autowired
118
    private SchemeItemRepository schemeItemRepository;
119
    @Autowired
120
    private SchemeRegionRepository schemeRegionRepository;
121
    @Autowired
122
    private WalletService walletService;
123
    @Autowired
124
    private PurchaseRepository purchaseRepository;
125
    @Autowired
126
    private FofoOrderRepository fofoOrderRepository;
31921 amit.gupta 127
    @Autowired
128
    private PriceCircularService priceCircularService;
33600 ranu 129
    @Autowired
130
    private SamsungExceptionRepository samsungExceptionRepository;
131
 
31410 amit.gupta 132
    @Override
34221 tejus.loha 133
    public List<String> getBlockedImeis() {
34254 tejus.loha 134
        List<SchemeBlockedImei> schemeBlockedImeis = schemeBlockedImeiRepository.selectAll();
34221 tejus.loha 135
        return schemeBlockedImeis.stream().filter(x -> x.isBlocked()).map(x -> x.getImei()).collect(Collectors.toList());
136
    }
137
 
138
    @Override
31410 amit.gupta 139
    public void saveScheme(int creatorId, CreateSchemeRequest createSchemeRequest) throws ProfitMandiBusinessException {
23444 amit.gupta 140
 
31410 amit.gupta 141
        this.validateCreateSchemeRequest(createSchemeRequest);
27898 amit.gupta 142
 
31410 amit.gupta 143
        Scheme scheme = this.toScheme(creatorId, createSchemeRequest);
25503 amit.gupta 144
 
31410 amit.gupta 145
        if (scheme.getStartDateTime().isAfter(scheme.getEndDateTime())) {
146
            throw new ProfitMandiBusinessException(
147
                    ProfitMandiConstants.START_DATE + ", " + ProfitMandiConstants.END_DATE,
148
                    scheme.getStartDateTime() + ", " + scheme.getEndDateTime(), "SCHM_VE_1005");
149
        }
26722 amit.gupta 150
 
31410 amit.gupta 151
        // this.validateItemIds(createSchemeRequest);
152
        schemeRepository.persist(scheme);
153
        for (int catalogId : createSchemeRequest.getCatalogIds()) {
154
            SchemeItem schemeItem = new SchemeItem();
155
            schemeItem.setSchemeId(scheme.getId());
156
            schemeItem.setCatalogId(catalogId);
34317 amit.gupta 157
            schemeItem.setStartDate(createSchemeRequest.getStartDate());
158
            schemeItem.setEndDate(createSchemeRequest.getEndDate());
31410 amit.gupta 159
            schemeItemRepository.persist(schemeItem);
160
        }
23444 amit.gupta 161
 
31410 amit.gupta 162
        for (int regionId : createSchemeRequest.getRegionIds()) {
163
            SchemeRegion schemeRegion = new SchemeRegion();
164
            schemeRegion.setSchemeId(scheme.getId());
165
            schemeRegion.setRegionId(regionId);
166
            schemeRegionRepository.persist(schemeRegion);
167
        }
26332 amit.gupta 168
 
31410 amit.gupta 169
    }
26722 amit.gupta 170
 
31410 amit.gupta 171
    private void validateCreateSchemeRequest(CreateSchemeRequest createSchemeRequest)
172
            throws ProfitMandiBusinessException {
173
        if (createSchemeRequest.getName() == null || createSchemeRequest.getName().isEmpty()) {
174
            throw new ProfitMandiBusinessException(ProfitMandiConstants.NAME, createSchemeRequest.getName(),
175
                    "SCHM_VE_1000");
176
        }
177
        if (createSchemeRequest.getAmount() <= 0) {
178
            throw new ProfitMandiBusinessException(ProfitMandiConstants.AMOUNT, createSchemeRequest.getAmount(),
179
                    "SCHM_VE_1001");
180
        }
26684 amit.gupta 181
 
31410 amit.gupta 182
        if (createSchemeRequest.getAmountType().equals(AmountType.PERCENTAGE)
183
                && createSchemeRequest.getAmount() > 100) {
184
            throw new ProfitMandiBusinessException(ProfitMandiConstants.AMOUNT, createSchemeRequest.getAmount(),
185
                    "SCHM_VE_1002");
186
        }
23444 amit.gupta 187
 
31410 amit.gupta 188
        if (createSchemeRequest.getStartDate() == null) {
189
            throw new ProfitMandiBusinessException(ProfitMandiConstants.START_DATE, createSchemeRequest.getStartDate(),
190
                    "SCHM_VE_1003");
191
        }
192
        if (createSchemeRequest.getEndDate() == null) {
193
            throw new ProfitMandiBusinessException(ProfitMandiConstants.END_DATE, createSchemeRequest.getEndDate(),
194
                    "SCHM_VE_1004");
195
        }
196
    }
23798 amit.gupta 197
 
31410 amit.gupta 198
    private void validateItemIds(CreateSchemeRequest createSchemeRequest) throws ProfitMandiBusinessException {
199
        if (createSchemeRequest.getCatalogIds() == null || createSchemeRequest.getCatalogIds().isEmpty()) {
200
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ITEM_ID, createSchemeRequest.getCatalogIds(),
201
                    "SCHM_1003");
202
        }
203
        List<Integer> foundItemIds = itemRepository.selectIdsByIdsAndType(createSchemeRequest.getCatalogIds(),
204
                ItemType.SERIALIZED);
205
        if (foundItemIds.size() != createSchemeRequest.getCatalogIds().size()) {
206
            createSchemeRequest.getCatalogIds().removeAll(foundItemIds);
207
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ITEM_ID, createSchemeRequest.getCatalogIds(),
208
                    "SCHM_1004");
209
        }
210
    }
29927 amit.gupta 211
 
31410 amit.gupta 212
    @Override
213
    public Scheme getSchemeById(int schemeId) throws ProfitMandiBusinessException {
214
        Scheme scheme = schemeRepository.selectById(schemeId);
215
        List<Integer> catalogIds = schemeItemRepository.selectCatalogIdsBySchemeId(scheme.getId());
216
        if (catalogIds.size() > 0) {
217
            List<Item> items = itemRepository.selectAllByCatalogIds(new HashSet<>(catalogIds));
218
            scheme.setCatalogStringMap(this.toCatalogStringMap(items));
219
        }
220
        return scheme;
221
    }
24562 amit.gupta 222
 
31410 amit.gupta 223
    public Map<Integer, String> toCatalogStringMap(List<Item> items) {
224
        Map<Integer, String> catalogMap = new HashMap<>();
225
        for (Item item : items) {
226
            if (!catalogMap.containsKey(item.getCatalogItemId())) {
227
                catalogMap.put(item.getCatalogItemId(), item.getItemDescriptionNoColor());
228
            }
229
        }
230
        return catalogMap;
231
    }
29927 amit.gupta 232
 
31410 amit.gupta 233
    private Set<Integer> schemeItemsToCatalogIds(List<SchemeItem> schemeItems) {
234
        Set<Integer> catalogId = new HashSet<>();
235
        for (SchemeItem schemeItem : schemeItems) {
236
            catalogId.add(schemeItem.getCatalogId());
237
        }
238
        return catalogId;
239
    }
23444 amit.gupta 240
 
31410 amit.gupta 241
    @Override
33248 ranu 242
    public List<SchemeModel> getAllSchemeModels(LocalDateTime startDateTime, LocalDateTime endDateTime) throws ProfitMandiBusinessException {
31410 amit.gupta 243
        List<Scheme> schemes = schemeRepository.selectAllBetweenCreateTimestamp(startDateTime, endDateTime);
244
        Map<Integer, Scheme> schemeIdSchemeMap = schemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
245
        List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIds(schemeIdSchemeMap.keySet());
246
        Set<Integer> catalogIds = schemeItems.stream().map(x -> x.getCatalogId()).collect(Collectors.toSet());
247
        List<Item> items = itemRepository.selectAllByCatalogIds(catalogIds);
248
        Map<Integer, String> catalogStringMap = this.toCatalogStringMap(items);
249
        this.addCatalogIdsToSchemes(schemeItems, schemeIdSchemeMap, catalogStringMap);
250
        return this.toSchemeModels(schemeIdSchemeMap);
251
    }
23444 amit.gupta 252
 
31410 amit.gupta 253
    private List<SchemeModel> toSchemeModels(Map<Integer, Scheme> schemeIdSchemeMap) {
254
        List<SchemeModel> schemeModels = new ArrayList<>();
255
        for (Map.Entry<Integer, Scheme> schemeIdSchemeEntry : schemeIdSchemeMap.entrySet()) {
256
            schemeModels.add(this.toSchemeModel(schemeIdSchemeEntry.getValue()));
257
        }
258
        return schemeModels;
259
    }
23444 amit.gupta 260
 
31410 amit.gupta 261
    private SchemeModel toSchemeModel(Scheme scheme) {
262
        SchemeModel schemeModel = new SchemeModel();
263
        schemeModel.setSchemeId(scheme.getId());
264
        schemeModel.setName(scheme.getName());
265
        schemeModel.setDescription(scheme.getDescription());
266
        schemeModel.setSchemeType(scheme.getType().toString());
267
        schemeModel.setAmountType(scheme.getAmountType().toString());
268
        schemeModel.setAmount(scheme.getAmount());
269
        schemeModel.setStartDateTime(StringUtils.toString(scheme.getStartDateTime()));
270
        schemeModel.setEndDateTime(StringUtils.toString(scheme.getEndDateTime()));
271
        schemeModel.setCreateTimestamp(StringUtils.toString(scheme.getCreateTimestamp()));
272
        schemeModel.setActiveTimestamp(StringUtils.toString(scheme.getActiveTimestamp()));
273
        schemeModel.setExpireTimestamp(StringUtils.toString(scheme.getExpireTimestamp()));
274
        schemeModel.setCreatedBy(scheme.getCreatedBy());
275
        schemeModel.setCatalogStringMap(scheme.getCatalogStringMap());
276
        schemeModel.setRetailerIds(scheme.getRetailerIds());
277
        return schemeModel;
278
    }
23444 amit.gupta 279
 
31410 amit.gupta 280
    private void addCatalogIdsToSchemes(List<SchemeItem> schemeItems, Map<Integer, Scheme> schemeIdSchemeMap,
281
                                        Map<Integer, String> catalogStringMap) {
282
        for (SchemeItem schemeItem : schemeItems) {
283
            Scheme scheme = schemeIdSchemeMap.get(schemeItem.getSchemeId());
284
            scheme.getCatalogStringMap().put(schemeItem.getCatalogId(), catalogStringMap.get(schemeItem.getCatalogId()));
285
        }
286
    }
31170 amit.gupta 287
 
31410 amit.gupta 288
    @Override
289
    public void activeSchemeById(int schemeId) throws ProfitMandiBusinessException {
290
        Scheme scheme = schemeRepository.selectById(schemeId);
291
        if (scheme.getActiveTimestamp() != null) {
292
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ACTIVE_TIMESTAMP, scheme.getActiveTimestamp(),
293
                    "SCHM_1005");
294
        }
295
        if (scheme.getExpireTimestamp() != null) {
296
            throw new ProfitMandiBusinessException(ProfitMandiConstants.EXPIRE_TIMESTAMP, scheme.getExpireTimestamp(),
297
                    "SCHM_1006");
298
        }
299
        scheme.setActiveTimestamp(LocalDateTime.now());
300
        this.sendSchemeNotification(scheme);
23444 amit.gupta 301
 
302
 
31410 amit.gupta 303
        /*
304
         * if (scheme.getType() == SchemeType.IN) {
305
         * this.processPreviousPurchases(scheme); } else if (scheme.getType() ==
306
         * SchemeType.OUT) { this.processPreviousSales(scheme); }
307
         */
308
    }
23796 amit.gupta 309
 
34568 vikas.jang 310
    @Override
311
    public void activeSchemeByIds(List<Scheme> schemes) throws ProfitMandiBusinessException {
312
        if (schemes.size() > 0) {
313
            for (Scheme scheme : schemes) {
314
                if (scheme.getActiveTimestamp() != null || scheme.getExpireTimestamp() != null) {
315
                    continue;
316
                }
317
                scheme.setActiveTimestamp(LocalDateTime.now());
318
            }
319
            this.sendCombinedSchemesNotification(schemes);
320
        }
321
    }
322
 
35252 aman 323
    @Override
324
    public void expireSchemeByIds(List<Scheme> schemes) throws ProfitMandiBusinessException {
325
        if (schemes.size() > 0) {
326
            for (Scheme scheme : schemes) {
327
                if (scheme.getExpireTimestamp() != null) {
328
                    continue;
329
                }
330
                scheme.setExpireTimestamp(scheme.getEndDateTime());
331
            }
35334 amit 332
            //this.sendCombinedSchemesNotification(schemes);
35252 aman 333
        }
334
    }
335
 
31410 amit.gupta 336
    private void sendSchemeNotification(Scheme scheme) throws ProfitMandiBusinessException {
35060 amit 337
        if (NOFITY_SCHEME_TYPES.contains(scheme.getType())) {
31336 amit.gupta 338
 
31410 amit.gupta 339
            SendNotificationModel sendNotificationModel = new SendNotificationModel();
340
            List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIds(Collections.singleton(scheme.getId()));
341
            Set<Integer> catalogIds = schemeItems.stream().map(x -> x.getCatalogId()).collect(Collectors.toSet());
342
            List<String> itemDescriptions = itemRepository.selectAllByCatalogIds(catalogIds).stream().filter(Utils.distinctByKey(Item::getCatalogItemId))
343
                    .map(x -> x.getItemDescriptionNoColor()).collect(Collectors.toList());
35210 vikas 344
            String titleTemplate = "Important Notice!\n\n %s of Rs.%s for %s";
31410 amit.gupta 345
            String schemeString = "Activation scheme";
346
            if (scheme.getType().equals(SchemeType.SPECIAL_SUPPORT)) {
347
                schemeString = "Special Support";
348
            }
30957 amit.gupta 349
 
31410 amit.gupta 350
            String message = "Duration from - " + FormattingUtils.formatDateMonth(scheme.getStartDateTime()) + " - " + FormattingUtils.formatDateMonth(scheme.getEndDateTime());
351
            sendNotificationModel.setCampaignName("activationscheme");
34902 vikas 352
            sendNotificationModel.setUrl("https://smartdukaan.com/pages/home/notifications");
31410 amit.gupta 353
            sendNotificationModel.setExpiresat(LocalDateTime.now().plusDays(1));
354
            sendNotificationModel.setMessage(message);
355
            sendNotificationModel.setTitle(String.format(titleTemplate, schemeString, FormattingUtils.formatDecimal(scheme.getAmount()), org.apache.commons.lang3.StringUtils.abbreviate(String.join(", ", itemDescriptions), 25)));
356
            sendNotificationModel.setType("url");
357
            sendNotificationModel.setMessageType(MessageType.scheme);
35060 amit 358
            List<SchemeRegion> schemeRegionList = schemeRegionRepository.selectAllBySchemeIds(Arrays.asList(scheme.getId()));
359
            sendNotificationModel.setRegionIds(schemeRegionList.stream().map(x->x.getRegionId()).collect(Collectors.toList()));
35236 amit 360
            notificationService.sendNotification(sendNotificationModel);
31410 amit.gupta 361
        }
362
    }
23444 amit.gupta 363
 
34709 amit.gupta 364
    @Autowired
365
    LoanRepository loanRepository;
34708 ranu 366
 
31410 amit.gupta 367
    @Override
368
    public void expireSchemeById(int schemeId, LocalDateTime expiryTime) throws ProfitMandiBusinessException {
369
        Scheme scheme = schemeRepository.selectById(schemeId);
370
        if (scheme == null || scheme.getActiveTimestamp() == null) {
371
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ACTIVE_TIMESTAMP, scheme.getActiveTimestamp(),
372
                    "SCHM_1007");
373
        }
374
        if (scheme.getExpireTimestamp() != null) {
375
            throw new ProfitMandiBusinessException(ProfitMandiConstants.EXPIRE_TIMESTAMP, scheme.getExpireTimestamp(),
376
                    "SCHM_1008");
377
        }
378
        scheme.setExpireTimestamp(LocalDateTime.now());
379
        if (expiryTime.isAfter(scheme.getEndDateTime())) {
380
            throw new ProfitMandiBusinessException(ProfitMandiConstants.EXPIRE_TIMESTAMP, scheme.getExpireTimestamp(),
381
                    "End Date cant be extended during expiry");
382
        }
36490 amit 383
        LocalDateTime originalEndDate = scheme.getEndDateTime();
31410 amit.gupta 384
        scheme.setEndDateTime(expiryTime);
36490 amit 385
 
386
        List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeId(schemeId);
387
        for (SchemeItem schemeItem : schemeItems) {
388
            if (schemeItem.getEndDate().equals(originalEndDate)) {
389
                schemeItem.setEndDate(expiryTime);
390
            }
391
        }
31410 amit.gupta 392
    }
23444 amit.gupta 393
 
31410 amit.gupta 394
    private Map<Integer, Scheme> toSchemeIdSchemeMap(List<Scheme> schemes) {
395
        Map<Integer, Scheme> schemeIdSchemeMap = new HashMap<>();
396
        for (Scheme scheme : schemes) {
397
            schemeIdSchemeMap.put(scheme.getId(), scheme);
398
        }
399
        return schemeIdSchemeMap;
400
    }
23444 amit.gupta 401
 
31410 amit.gupta 402
    private Map<Integer, Set<Scheme>> toCatalogIdSchemesMap(List<SchemeItem> schemeItems, List<Scheme> schemes) {
403
        Map<Integer, Scheme> schemesMap = schemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
404
        Map<Integer, Set<Scheme>> catalogSchemesMap = new HashMap<>();
405
        for (SchemeItem schemeItem : schemeItems) {
406
            if (!catalogSchemesMap.containsKey(schemeItem.getCatalogId())) {
407
                catalogSchemesMap.put(schemeItem.getCatalogId(), new HashSet<>());
408
            }
409
            Set<Scheme> schemesSet = catalogSchemesMap.get(schemeItem.getCatalogId());
410
            schemesSet.add(schemesMap.get(schemeItem.getSchemeId()));
411
        }
412
        return catalogSchemesMap;
413
    }
23444 amit.gupta 414
 
31410 amit.gupta 415
    private Map<InventoryItem, Set<Scheme>> toInventoryItemSchemesMap(List<Scheme> schemes,
33248 ranu 416
                                                                      List<InventoryItem> inventoryItems) throws ProfitMandiBusinessException {
31410 amit.gupta 417
        Set<Integer> schemeIds = schemes.stream().map(x -> x.getId()).collect(Collectors.toSet());
418
        Set<Integer> itemIds = inventoryItems.stream().map(x -> x.getItemId()).collect(Collectors.toSet());
419
        Set<Integer> catalogIds = itemRepository.selectByIds(itemIds).stream().map(x -> x.getCatalogItemId()).collect(Collectors.toSet());
420
        List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIdsAndCatalogIds(schemeIds, catalogIds);
23444 amit.gupta 421
 
31410 amit.gupta 422
        Map<Integer, Set<Scheme>> catalogIdSchemesMap = this.toCatalogIdSchemesMap(schemeItems, schemes);
423
        Map<InventoryItem, Set<Scheme>> inventoryItemSchemesMap = new HashMap<>();
424
        for (InventoryItem inventoryItem : inventoryItems) {
425
            LOGGER.info("inventoryItem {}", inventoryItem);
426
            LOGGER.info("inventoryItem.getItem() {}", inventoryItem.getItem());
427
            LOGGER.info("catalogIdSchemesMap {}", catalogIdSchemesMap);
428
            if (catalogIdSchemesMap.containsKey(inventoryItem.getItem().getCatalogItemId())) {
429
                inventoryItemSchemesMap.put(inventoryItem, catalogIdSchemesMap.get(inventoryItem.getItem().getCatalogItemId()));
430
            }
431
        }
432
        return inventoryItemSchemesMap;
433
    }
31170 amit.gupta 434
 
31903 amit.gupta 435
    @Autowired
436
    OfferTargetSlabRepository offerTargetSlabRepository;
437
 
31410 amit.gupta 438
    private Scheme toScheme(int creatorId, CreateSchemeRequest createSchemeRequest) {
439
        Scheme scheme = new Scheme();
440
        scheme.setName(createSchemeRequest.getName());
441
        scheme.setDescription(createSchemeRequest.getDescription());
442
        scheme.setType(createSchemeRequest.getType());
443
        scheme.setAmountType(createSchemeRequest.getAmountType());
444
        scheme.setAmount(createSchemeRequest.getAmount());
445
        scheme.setPartnerType(createSchemeRequest.getPartnerType());
446
        scheme.setStartDateTime(createSchemeRequest.getStartDate());
447
        scheme.setEndDateTime(createSchemeRequest.getEndDate());
448
        scheme.setCreatedBy(creatorId);
449
        scheme.setCashback(createSchemeRequest.isCashback());
36451 amit 450
        scheme.setReference(createSchemeRequest.getReference());
31410 amit.gupta 451
        return scheme;
452
    }
23444 amit.gupta 453
 
31903 amit.gupta 454
    @Autowired
455
    OfferRepository offerRepository;
456
    @Autowired
457
    OfferPayoutRepository offerPayoutRepository;
458
    @Autowired
459
    AgeingService ageingService;
31588 amit.gupta 460
 
35060 amit 461
    @Override
462
    public void sendCombinedSchemesNotification(List<Scheme> schemes) throws ProfitMandiBusinessException {
34709 amit.gupta 463
        if (schemes == null || schemes.isEmpty()) return;
464
 
465
 
35060 amit 466
        List<Scheme> filteredSchemes = schemes.stream()
467
                .filter(s -> NOFITY_SCHEME_TYPES.contains(s.getType())).collect(Collectors.toList());
34709 amit.gupta 468
 
35060 amit 469
        Map<Integer, Scheme> schemesMap = filteredSchemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
34709 amit.gupta 470
 
35060 amit 471
        Map<Integer, List<Integer>> schemeIdRegionIdsMap = schemeRegionRepository.selectAllBySchemeIds(new ArrayList<>(schemesMap.keySet()))
472
                .stream().collect(Collectors.groupingBy(SchemeRegion::getSchemeId,
473
                        Collectors.collectingAndThen(Collectors.mapping(SchemeRegion::getRegionId, Collectors.toList()),
474
                                list -> {
475
                                    list.sort(Integer::compareTo); // sort ascending
476
                                    return list;
477
                                })));
34709 amit.gupta 478
 
35394 amit 479
        // Batch fetch all schemeItems for all filtered schemes at once
480
        Map<Integer, List<SchemeItem>> schemeItemsBySchemeId = schemeItemRepository.selectBySchemeIds(schemesMap.keySet())
481
                .stream().collect(Collectors.groupingBy(SchemeItem::getSchemeId));
482
 
483
        // Collect all catalogIds for batch fetching items
484
        Set<Integer> allCatalogIdsForFetch = schemeItemsBySchemeId.values().stream()
485
                .flatMap(List::stream)
486
                .map(SchemeItem::getCatalogId)
487
                .collect(Collectors.toSet());
488
 
489
        // Batch fetch all items by catalogIds once
490
        Map<Integer, List<Item>> itemsByCatalogId = allCatalogIdsForFetch.isEmpty()
491
                ? Collections.emptyMap()
492
                : itemRepository.selectAllByCatalogIds(allCatalogIdsForFetch).stream()
493
                        .collect(Collectors.groupingBy(Item::getCatalogItemId));
494
 
35060 amit 495
        Map<List<Integer>, Map<String, List<Scheme>>> groupedByRegionDate = filteredSchemes.stream()
496
                .collect(Collectors.groupingBy(s -> schemeIdRegionIdsMap.get(s.getId()), Collectors.groupingBy(s ->
497
                        String.valueOf(s.getStartDateTime()).substring(0, 10) + " - " +
498
                                String.valueOf(s.getEndDateTime()).substring(0, 10)
499
                )));
34709 amit.gupta 500
 
35060 amit 501
        for (Map.Entry<List<Integer>, Map<String, List<Scheme>>> regionIdsDateKeyset : groupedByRegionDate.entrySet()) {
502
            SendNotificationModel sendNotificationModel = new SendNotificationModel();
503
            StringBuilder messageBuilder = new StringBuilder();
504
            Set<Integer> allCatalogIds = new HashSet<>();
505
            Set<String> allBrands = new HashSet<>();
506
            String schemeTypeLabel = null;
507
            List<Integer> regionIds = regionIdsDateKeyset.getKey();
508
            for (Map.Entry<String, List<Scheme>> entry : regionIdsDateKeyset.getValue().entrySet()) {
509
                String dateRange = entry.getKey();
510
                List<Scheme> groupedSchemes = entry.getValue();
34709 amit.gupta 511
 
512
 
35060 amit 513
                messageBuilder.append(String.format("Duration: %s\n", dateRange));
34962 vikas 514
 
35060 amit 515
                for (Scheme scheme : groupedSchemes) {
516
                    if (scheme.getType().equals(SchemeType.SPECIAL_SUPPORT)) {
517
                        schemeTypeLabel = "Special Support";
518
                    } else if (scheme.getType().equals(SchemeType.SELLOUT)) {
519
                        schemeTypeLabel = "Sellout Scheme";
520
                    } else if (scheme.getType().equals(SchemeType.SELLIN)) {
521
                        schemeTypeLabel = "Sellin Scheme";
522
                    }
34962 vikas 523
 
35394 amit 524
                    // Use pre-fetched schemeItems instead of N+1 query
525
                    List<SchemeItem> schemeItems = schemeItemsBySchemeId.getOrDefault(scheme.getId(), Collections.emptyList());
35060 amit 526
 
527
                    Set<Integer> catalogIds = schemeItems.stream()
528
                            .map(SchemeItem::getCatalogId)
529
                            .collect(Collectors.toSet());
530
                    allCatalogIds.addAll(catalogIds);
531
 
35394 amit 532
                    // Use pre-fetched items instead of duplicate DB calls
533
                    List<Item> catalogItems = catalogIds.stream()
534
                            .flatMap(cid -> itemsByCatalogId.getOrDefault(cid, Collections.emptyList()).stream())
535
                            .collect(Collectors.toList());
536
 
537
                    List<String> itemBrand = catalogItems.stream()
35060 amit 538
                            .map(Item::getBrand)
539
                            .filter(Objects::nonNull)
540
                            .distinct()
541
                            .collect(Collectors.toList());
542
                    allBrands.addAll(itemBrand);
543
 
35394 amit 544
                    List<String> itemDescriptions = catalogItems.stream()
35060 amit 545
                            .filter(Utils.distinctByKey(Item::getCatalogItemId))
546
                            .map(Item::getItemDescriptionNoColor)
547
                            .collect(Collectors.toList());
548
 
549
                    messageBuilder.append(String.format("%s of Rs.%s on Models: %s\n",
550
                            schemeTypeLabel,
551
                            FormattingUtils.formatDecimal(scheme.getAmount()),
552
                            String.join(", ", itemDescriptions)));
553
                }
554
                messageBuilder.append("\n");
34962 vikas 555
            }
34709 amit.gupta 556
 
35060 amit 557
            if (allBrands.isEmpty() || schemeTypeLabel == null) return;
34709 amit.gupta 558
 
35060 amit 559
            String title = String.format("%s for %s",
560
                    schemeTypeLabel,
561
                    String.join(", ", allBrands));
34962 vikas 562
 
35060 amit 563
            sendNotificationModel.setCampaignName("activationscheme");
564
            sendNotificationModel.setUrl("https://store.smartdukaan.com/pages/home/notifications");
565
            sendNotificationModel.setExpiresat(LocalDateTime.now().plusDays(1));
566
            sendNotificationModel.setTitle(title);
567
            sendNotificationModel.setMessage(messageBuilder.toString().trim());
568
            sendNotificationModel.setType("url");
569
            sendNotificationModel.setMessageType(MessageType.scheme);
570
            sendNotificationModel.setRegionIds(regionIds);
571
            System.out.println(sendNotificationModel);
572
            notificationService.sendNotification(sendNotificationModel);
573
        }
574
 
34709 amit.gupta 575
    }
576
 
577
    //CDs would be rejected based on cdFreeDays
31903 amit.gupta 578
    public void processSchemeIn(int purchaseId, int retailerId) throws ProfitMandiBusinessException {
34504 amit.gupta 579
        LOGGER.info("Processing scheme in  for purchaseId - {}", purchaseId);
31903 amit.gupta 580
        Purchase purchase = purchaseRepository.selectByIdAndFofoId(purchaseId, retailerId);
34709 amit.gupta 581
        String purchaseInvoice = purchase.getPurchaseReference();
582
 
583
        Loan loan = loanRepository.selectLoanByInvoice(purchaseInvoice);
584
 
34317 amit.gupta 585
        PartnerType partnerType = partnerTypeChangeService.getTypeOnMonth(retailerId,
586
                YearMonth.from(purchase.getCreateTimestamp()));
34139 amit.gupta 587
        List<InventoryItem> inventoryItems = inventoryItemRepository.selectByPurchaseId(purchaseId);
31903 amit.gupta 588
        //Remove imeis from blocked imeis list
34317 amit.gupta 589
        List<String> blockedImeis = this.getBlockedImeis();
590
        inventoryItems = inventoryItems.stream().filter(inventoryItem -> !blockedImeis.contains(inventoryItem.getSerialNumber())).collect(Collectors.toList());
31903 amit.gupta 591
        if (inventoryItems.size() == 0) return;
23444 amit.gupta 592
 
31903 amit.gupta 593
        Set<Integer> itemIds = inventoryItems.stream().map(x -> x.getItemId()).collect(Collectors.toSet());
594
        Map<Integer, Item> itemsMap = itemRepository.selectByIds(itemIds).stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
595
        inventoryItems.stream().forEach(x -> x.setItem(itemsMap.get(x.getItemId())));
23444 amit.gupta 596
 
597
 
34317 amit.gupta 598
        LocalDateTime billingDate = purchaseService.getBillingDateOfPurchase(purchaseId);
31903 amit.gupta 599
        Set<Integer> itemIdsSet = tagListingRepository.selectByItemIdsAndTagIds(itemIds, tagIds).stream()
600
                .filter(x -> x.getEolDate() == null || x.getEolDate().isAfter(billingDate)).map(x -> x.getItemId())
601
                .collect(Collectors.toSet());
602
        // Only consider inventory items that were not returned and not eol
603
        inventoryItems = inventoryItems.stream().filter(x -> itemIdsSet.contains(x.getItemId()))
604
                .filter(x -> !x.getLastScanType().equals(ScanType.PURCHASE_RET_BAD))
605
                .filter(x -> !x.getLastScanType().equals(ScanType.PURCHASE_RET)).collect(Collectors.toList());
606
 
607
        if (inventoryItems.size() == 0) return;
608
 
609
        Map<Integer, List<InventoryItem>> catalogInventoryItemMap = inventoryItems.stream().collect(Collectors.groupingBy(x -> x.getItem().getCatalogItemId()));
34317 amit.gupta 610
        Map<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSchemeSummaryMap = tagListingRepository.getModelSchemesByCatalogIdsAndType(retailerId,
32454 amit.gupta 611
                partnerType, new ArrayList<>(catalogInventoryItemMap.keySet()), billingDate);
31903 amit.gupta 612
 
613
 
614
        int itemsCount = 0;
615
        float totalCashback = 0;
34709 amit.gupta 616
        //LOGGER.info("catalogSchemeSummaryMap - {}", catalogSchemeSummaryMap);
31903 amit.gupta 617
        for (Map.Entry<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSummaryModelListEntry : catalogSchemeSummaryMap.entrySet()) {
618
            CatalogSummaryModel catalogSummaryModel = catalogSummaryModelListEntry.getKey();
34504 amit.gupta 619
            List<SchemeSummaryModel> schemeSummaryModels = catalogSummaryModelListEntry.getValue().stream().filter(Objects::nonNull).collect(Collectors.toList());
34317 amit.gupta 620
            schemeSummaryModels.stream().filter(x -> x != null && x.getSchemeType().getTransactionType().equals(StockTransactionType.IN)).forEach(x -> x.setProcess(true));
31903 amit.gupta 621
            if (schemeSummaryModels.stream().filter(x -> x.isProcess()).count() == 0) continue;
622
            List<InventoryItem> modelInventoryItems = catalogInventoryItemMap.get(catalogSummaryModel.getCatalogId());
623
            for (InventoryItem inventoryItem : modelInventoryItems) {
36405 amit 624
                inventoryItemRepository.selectForUpdate(inventoryItem.getId());
32165 amit.gupta 625
                float inventoryItemCashback = this.createSchemeInOut(schemeSummaryModels, inventoryItem);
34709 amit.gupta 626
                if (loan != null && loan.getCdFreeDays() > 0) {
627
                    List<SchemeType> rejectedCDTypes = HALF_DAYS_CD_REJECT_SCHEME_TYPES;
36070 amit 628
                    if (loan.getCdFreeDays() == ProfitMandiConstants.LOAN_FULL_CREDIT_DAYS
629
                            || loan.getCdFreeDays() == ProfitMandiConstants.PREMIUM_LOAN_FULL_CREDIT_DAYS) {
34709 amit.gupta 630
                        rejectedCDTypes = FULL_DAYS_CD_SCHEME_TYPES;
631
                    }
632
                    for (SchemeType rejectedCDType : rejectedCDTypes) {
35394 amit 633
                        // Fixed: single DB call instead of duplicate calls
634
                        List<SchemeInOut> cdSios = schemeInOutRepository.selectAllByType(rejectedCDType, inventoryItem.getId());
635
                        SchemeInOut sio = cdSios.isEmpty() ? null : cdSios.get(0);
34800 amit.gupta 636
                        if (sio == null) continue;
34709 amit.gupta 637
                        sio.setStatus(SchemePayoutStatus.REJECTED);
34850 amit 638
                        //Noone should change this
34709 amit.gupta 639
                        sio.setStatusDescription("Rejected due to free days availed");
640
                        sio.setRolledBackTimestamp(LocalDateTime.now());
641
                        inventoryItemCashback -= sio.getAmount();
642
                    }
643
 
644
                }
645
                if (inventoryItemCashback > 0.01f) {
31903 amit.gupta 646
                    itemsCount++;
647
                    totalCashback += inventoryItemCashback;
31410 amit.gupta 648
                }
649
            }
31903 amit.gupta 650
        }
22653 ashik.ali 651
 
29927 amit.gupta 652
 
34317 amit.gupta 653
        LOGGER.info("Items count for purchase id {} is {}", purchaseId, itemsCount);
31903 amit.gupta 654
        if (itemsCount > 0) {
655
            walletService.addAmountToWallet(
34317 amit.gupta 656
                    retailerId, purchaseId, WalletReferenceType.SCHEME_IN, "Added for SCHEME IN against invoice "
31903 amit.gupta 657
                            + purchase.getPurchaseReference() + " (total " + itemsCount + " pcs)",
658
                    totalCashback, purchase.getCreateTimestamp());
659
            LOGGER.info("Added Rs.{} for SCHEME IN against invoice {} total pcs({}) {}", totalCashback,
660
                    purchase.getPurchaseReference(), itemsCount);
34504 amit.gupta 661
        }
34317 amit.gupta 662
 
31903 amit.gupta 663
    }
664
 
665
 
35895 amit 666
    // Calculate payout amount for an inventory item based on scheme type.
667
    // For PERCENTAGE schemes: uses price-drop-adjusted DP (purchase price minus any price drop).
668
    //   GRN (IN): dpForCalc = purchasePrice - priceDropAmount
669
    //   Sale (OUT): dpForCalc = min(purchasePrice - priceDropAmount, currentSellingPrice)
670
    //   On price increase, priceDropAmount stays 0 so payout is on original purchase price.
671
    // For FIXED schemes (IN only): returns the fixed scheme amount directly.
31410 amit.gupta 672
    private float getAmount(InventoryItem inventoryItem, Scheme scheme) throws ProfitMandiBusinessException {
34221 tejus.loha 673
        if (this.getBlockedImeis().contains(inventoryItem.getSerialNumber())) {
31410 amit.gupta 674
            return 0;
675
        }
676
        float amount = 0;
677
        float dpForCalc = 0;
29927 amit.gupta 678
 
31410 amit.gupta 679
        if (scheme.getAmountType().equals(AmountType.PERCENTAGE)) {
34317 amit.gupta 680
            if (scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
31410 amit.gupta 681
                dpForCalc = inventoryItem.getUnitPrice() - inventoryItem.getPriceDropAmount();
682
            } else {
683
                try {
684
                    dpForCalc = Math.min(inventoryItem.getUnitPrice() - inventoryItem.getPriceDropAmount(),
685
                            tagListingRepository.selectByItemId(inventoryItem.getItemId()).getSellingPrice());
686
                } catch (Exception e) {
35895 amit 687
                    LOGGER.info("Could not find tag Listing entry for itemId {}", inventoryItem.getItemId());
31410 amit.gupta 688
                }
689
            }
690
            amount = dpForCalc * scheme.getAmount() / 100;
35895 amit 691
            LOGGER.debug("Scheme payout: invId={}, imei={}, schemeId={}, dpForCalc={}, schemeAmt={}%, payout={}",
692
                    inventoryItem.getId(), inventoryItem.getSerialNumber(), scheme.getId(),
693
                    dpForCalc, scheme.getAmount(), amount);
34317 amit.gupta 694
        } else if (scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
31410 amit.gupta 695
            amount = scheme.getAmount();
696
        }
697
        return amount;
698
    }
23444 amit.gupta 699
 
31903 amit.gupta 700
    //Only in and activation margins are allowed to be rolled out more than twice
32165 amit.gupta 701
    private float createSchemeInOut(List<SchemeSummaryModel> schemeSummaryModels, InventoryItem inventoryItem) throws ProfitMandiBusinessException {
33614 amit.gupta 702
        LOGGER.info("schemeSummaryModels - {}", schemeSummaryModels);
31921 amit.gupta 703
        InventoryPayoutModel inventoryPayoutModel = priceCircularService.getPayouts(inventoryItem);
31903 amit.gupta 704
        //Get all schemes
705
        List<SchemeSummaryModel> inventoryPayoutModelToProcess = schemeSummaryModels.stream().filter(x -> x.isProcess()).collect(Collectors.toList());
34709 amit.gupta 706
        LOGGER.info("inventoryPayoutModel - {}", inventoryPayoutModel);
32165 amit.gupta 707
        List<SchemeInOut> paidSios = inventoryPayoutModel.getPaidSios();
708
        List<SchemeInOut> pendingSios = inventoryPayoutModel.getPendingSios();
709
        Map<SchemeType, List<SchemeInOut>> paidSchemeTypesMap = inventoryPayoutModel.getPaidSios().stream().collect(Collectors.groupingBy(x -> x.getScheme().getType()));
710
        Map<Integer, SchemeInOut> paidSchemesMap = paidSios.stream().collect(Collectors.toMap(x -> x.getSchemeId(), x -> x));
711
        Map<Integer, SchemeInOut> pendingSchemesMap = pendingSios.stream().collect(Collectors.toMap(x -> x.getSchemeId(), x -> x));
32155 amit.gupta 712
        Map<SchemeType, Float> schemeTypeCancelledAmountMap = new HashMap<>();
31903 amit.gupta 713
 
714
        double percentageToPay = 0d;
715
        double fixedToPay = 0d;
716
        Map<SchemeSummaryModel, AmountModel> payoutSchemeSummaryModelMap = new HashMap<>();
717
        for (SchemeSummaryModel schemeSummaryModelToProcess : inventoryPayoutModelToProcess) {
32972 amit.gupta 718
            if (paidSchemesMap.containsKey(schemeSummaryModelToProcess.getSchemeId()) || pendingSchemesMap.containsKey(schemeSummaryModelToProcess.getSchemeId()))
719
                continue;
34778 amit.gupta 720
            //All Valid margins are supposed to be credited invalid margins should be rejected
721
            if (SchemeType.SPECIAL_SUPPORT.equals(schemeSummaryModelToProcess.getSchemeType())) {
722
                //Create only if the activation date is not known
32165 amit.gupta 723
                ActivatedImei activatedImei = activatedImeiRepository.selectBySerialNumber(inventoryItem.getSerialNumber());
32972 amit.gupta 724
                if (activatedImei == null || activatedImei.getActivationTimestamp() == null) {
32165 amit.gupta 725
                    SchemeInOut sio = new SchemeInOut();
726
                    sio.setAmount(0);
727
                    sio.setInventoryItemId(inventoryItem.getId());
728
                    sio.setSchemeId(schemeSummaryModelToProcess.getSchemeId());
729
                    sio.setStatusDescription("Activation pending for IMEI#" + inventoryItem.getSerialNumber());
730
                    sio.setStatus(SchemePayoutStatus.PENDING);
731
                    schemeInOutRepository.persist(sio);
732
                }
34317 amit.gupta 733
            } else if (!StockTransactionType.IN.equals(schemeSummaryModelToProcess.getSchemeType().getTransactionType())) {
31903 amit.gupta 734
                //We have got non repeating scheme type
34709 amit.gupta 735
 
736
                if (schemeSummaryModelToProcess.getAmountType().equals(AmountType.PERCENTAGE)) {
737
                    percentageToPay += schemeSummaryModelToProcess.getAmount();
31903 amit.gupta 738
                } else {
34709 amit.gupta 739
                    fixedToPay += schemeSummaryModelToProcess.getAmount();
31903 amit.gupta 740
                }
34709 amit.gupta 741
                payoutSchemeSummaryModelMap.put(schemeSummaryModelToProcess, new AmountModel(schemeSummaryModelToProcess.getAmount(), schemeSummaryModelToProcess.getAmountType()));
742
 
31903 amit.gupta 743
            } else {
744
                if (schemeSummaryModelToProcess.getAmountType().equals(AmountType.PERCENTAGE)) {
34709 amit.gupta 745
                    //Check for rejected CashDiscounts dont continue if its ever rejected once
746
                    if (SchemeType.CDS.contains(schemeSummaryModelToProcess.getSchemeType())) {
747
                        List<SchemeInOut> cdSchemeInOuts = schemeInOutRepository.selectAllByType(schemeSummaryModelToProcess.getSchemeType(), inventoryItem.getId());
748
                        LOGGER.info(cdSchemeInOuts);
34850 amit 749
                        if (!cdSchemeInOuts.isEmpty() && cdSchemeInOuts.get(0).getStatusDescription().equals("Rejected due to free days availed"))
750
                            continue;
34709 amit.gupta 751
                    }
31903 amit.gupta 752
                    percentageToPay += schemeSummaryModelToProcess.getAmount();
753
                } else {
754
                    fixedToPay += schemeSummaryModelToProcess.getAmount();
755
                }
756
                payoutSchemeSummaryModelMap.put(schemeSummaryModelToProcess, new AmountModel(schemeSummaryModelToProcess.getAmount(), schemeSummaryModelToProcess.getAmountType()));
757
            }
758
        }
759
        double walletCredit = 0d;
760
        if (fixedToPay > 0) {
31913 amit.gupta 761
            double fixedRollout = fixedToPay * (100 / (100 + inventoryPayoutModel.getPercentageAmount()));
31903 amit.gupta 762
            for (Map.Entry<SchemeSummaryModel, AmountModel> schemeSummaryModelAmountModelEntry : payoutSchemeSummaryModelMap.entrySet()) {
763
                SchemeSummaryModel schemeSummaryModel = schemeSummaryModelAmountModelEntry.getKey();
764
                AmountModel amountModel = schemeSummaryModelAmountModelEntry.getValue();
765
                if (amountModel.getAmountType().equals(AmountType.FIXED)) {
766
                    SchemeInOut sio = new SchemeInOut();
767
                    sio.setSchemeId(schemeSummaryModel.getSchemeId());
768
                    sio.setInventoryItemId(inventoryItem.getId());
769
                    sio.setStatus(SchemePayoutStatus.CREDITED);
770
                    sio.setCreditTimestamp(LocalDateTime.now());
32155 amit.gupta 771
                    sio.setAmount((float) (fixedRollout * amountModel.getAmount() / fixedToPay) + schemeTypeCancelledAmountMap.getOrDefault(schemeSummaryModel.getSchemeType(), 0f));
34317 amit.gupta 772
                    if (schemeSummaryModel.getSchemeType().getTransactionType().equals(StockTransactionType.IN))
31903 amit.gupta 773
                        sio.setStatusDescription("Credited for GRN of IMEI-" + inventoryItem.getSerialNumber());
774
                    else
775
                        sio.setStatusDescription("Credited for Sale of IMEI-" + inventoryItem.getSerialNumber());
776
                    schemeInOutRepository.persist(sio);
777
                }
778
            }
779
            walletCredit += fixedRollout;
780
        }
781
 
782
        if (percentageToPay > 0) {
31987 amit.gupta 783
            LOGGER.info("inventoryPayoutModel.getFixedAmount() ----> {}", inventoryPayoutModel.getFixedAmount());
31914 amit.gupta 784
            double effectiveDP = inventoryPayoutModel.getDp() - (inventoryPayoutModel.getFixedAmount() + fixedToPay);
31903 amit.gupta 785
            double totalPercentage = inventoryPayoutModel.getPercentageAmount() + percentageToPay;
786
            double percentageRollout = effectiveDP * (totalPercentage / (100 + totalPercentage) - (inventoryPayoutModel.getPercentageAmount() / (100 + inventoryPayoutModel.getPercentageAmount())));
787
            for (Map.Entry<SchemeSummaryModel, AmountModel> schemeSummaryModelAmountModelEntry : payoutSchemeSummaryModelMap.entrySet()) {
788
                SchemeSummaryModel schemeSummaryModel = schemeSummaryModelAmountModelEntry.getKey();
789
                AmountModel amountModel = schemeSummaryModelAmountModelEntry.getValue();
790
                if (amountModel.getAmountType().equals(AmountType.PERCENTAGE)) {
791
                    SchemeInOut sio = new SchemeInOut();
792
                    sio.setInventoryItemId(inventoryItem.getId());
793
                    sio.setSchemeId(schemeSummaryModel.getSchemeId());
794
                    sio.setStatus(SchemePayoutStatus.CREDITED);
795
                    sio.setCreditTimestamp(LocalDateTime.now());
32155 amit.gupta 796
                    sio.setAmount((float) (percentageRollout * amountModel.getAmount() / percentageToPay) +
797
                            schemeTypeCancelledAmountMap.getOrDefault(schemeSummaryModel.getSchemeType(), 0f));
34317 amit.gupta 798
                    if (schemeSummaryModel.getSchemeType().getTransactionType().equals(StockTransactionType.IN))
31903 amit.gupta 799
                        sio.setStatusDescription("Credited for GRN of IMEI-" + inventoryItem.getSerialNumber());
800
                    else
801
                        sio.setStatusDescription("Credited for Sale of IMEI-" + inventoryItem.getSerialNumber());
802
                    schemeInOutRepository.persist(sio);
803
                }
804
            }
805
            walletCredit += percentageRollout;
806
        }
807
 
808
        return (float) walletCredit;
809
    }
810
 
811
    private Set<Integer> filterImeisByAgeing(Set<Integer> inventoryItemIds, FofoOrder fofoOrder) {
812
        Set<Integer> filteredInventoryIds = new HashSet<>();
813
        List<PartnerAgeingModel> partnerAgeingModels = ageingService.filterAgedInventory(inventoryItemIds);
35394 amit 814
 
815
        // Batch fetch all Samsung activated imeis at once to avoid N+1 queries
816
        List<String> samsungSerialNumbers = partnerAgeingModels.stream()
817
                .filter(p -> "Samsung".equalsIgnoreCase(p.getBrand()))
818
                .map(PartnerAgeingModel::getSerialNumber)
819
                .collect(Collectors.toList());
820
        Map<String, ActivatedImei> activatedImeiMap = samsungSerialNumbers.isEmpty()
821
                ? Collections.emptyMap()
822
                : activatedImeiRepository.selectBySerialNumbers(samsungSerialNumbers).stream()
823
                        .collect(Collectors.toMap(ActivatedImei::getSerialNumber, x -> x, (a, b) -> a));
824
 
31903 amit.gupta 825
        for (PartnerAgeingModel partnerAgeingModel : partnerAgeingModels) {
826
            LOGGER.info("Serial Number - {}", partnerAgeingModel.getSerialNumber());
827
            if (partnerAgeingModel.getBrand().equalsIgnoreCase("Samsung")) {
35394 amit 828
                ActivatedImei activatedImei = activatedImeiMap.get(partnerAgeingModel.getSerialNumber());
31903 amit.gupta 829
                if (activatedImei != null && activatedImei.getActivationTimestamp().toLocalDate()
830
                        .isBefore(partnerAgeingModel.getBillingDate().plusDays(partnerAgeingModel.getMaxAgeingDays()))) {
831
                    //Lets give money if activation is there and is before the ageing limit of that brand.
832
                    filteredInventoryIds.add(partnerAgeingModel.getInventoryItemId());
833
                } else {
834
                    //If billing happens before ageing expiry
835
                    if (fofoOrder.getCreateTimestamp().toLocalDate().isBefore(partnerAgeingModel.getBillingDate().plusDays(partnerAgeingModel.getMaxAgeingDays()))) {
836
                        filteredInventoryIds.add(partnerAgeingModel.getInventoryItemId());
837
                    }
838
                }
839
            } else {
840
                filteredInventoryIds.add(partnerAgeingModel.getInventoryItemId());
841
            }
842
        }
843
        return filteredInventoryIds;
844
    }
845
 
33087 amit.gupta 846
    @Autowired
847
    WarehouseInventoryItemRepository warehouseInventoryItemRepository;
848
 
31410 amit.gupta 849
    @Override
850
    public float processSchemeOut(int fofoOrderId, int retailerId) throws ProfitMandiBusinessException {
34746 amit.gupta 851
 
852
        float totalCashback = 0;
31410 amit.gupta 853
        FofoOrder fofoOrder = fofoOrderRepository.selectByFofoIdAndOrderId(retailerId, fofoOrderId);
854
        // Process only if order is not cancelled
855
        if (fofoOrder.getCancelledTimestamp() == null) {
856
            // PartnerType partnerType = partnerTypeChangeService.getTypeOnDate(retailerId,
857
            // fofoOrder.getCreateTimestamp().toLocalDate());
858
            // TODO - SCHEME
859
            PartnerType partnerType = partnerTypeChangeService.getTypeOnMonth(retailerId,
860
                    YearMonth.from(fofoOrder.getCreateTimestamp()));
34778 amit.gupta 861
            LOGGER.info("OrderDate - {}, Partner Type - {}", fofoOrder.getCreateTimestamp(), partnerType);
29927 amit.gupta 862
 
31410 amit.gupta 863
            List<ScanRecord> scanRecords = scanRecordRepository.selectAllByOrderId(fofoOrderId);
864
            if (scanRecords.size() == 0) return 0;
865
            Set<Integer> inventoryItemIds = scanRecords.stream().map(x -> x.getInventoryItemId())
866
                    .collect(Collectors.toSet());
31903 amit.gupta 867
            //Check for ageing
31912 amit.gupta 868
            //inventoryItemIds = this.filterImeisByAgeing(inventoryItemIds, fofoOrder);
31903 amit.gupta 869
            //ageingService.filterAgedInventory(inventoryItemIds);
31410 amit.gupta 870
            LOGGER.info("fofoOrderId --- {}", fofoOrderId);
871
            LOGGER.info("scanRecords --- {}", scanRecords);
872
            LOGGER.info("inventoryItemIds --- {}", inventoryItemIds);
31903 amit.gupta 873
 
874
            if (inventoryItemIds.size() == 0) return 0;
875
 
35394 amit 876
            // Cache blocked imeis to avoid repeated DB calls inside stream
877
            Set<String> blockedImeisSet = new HashSet<>(this.getBlockedImeis());
878
 
31410 amit.gupta 879
            Set<InventoryItem> inventoryItems = inventoryItemRepository.selectByIds(inventoryItemIds).stream()
880
                    .filter(x -> x.getSerialNumber() != null && !x.getSerialNumber().equals(""))
881
                    .collect(Collectors.toSet());
35394 amit 882
            inventoryItems = inventoryItems.stream().filter(inventoryItem -> !blockedImeisSet.contains(inventoryItem.getSerialNumber())).collect(Collectors.toSet());
29927 amit.gupta 883
 
33087 amit.gupta 884
            //Do not consider imei above 90 days for samsung
33432 amit.gupta 885
            List<String> samsungSerialNumbers = inventoryItems.stream().filter(x -> x.getItem().getBrand().equalsIgnoreCase("samsung")).map(x -> x.getSerialNumber()).collect(Collectors.toList());
33600 ranu 886
 
33432 amit.gupta 887
            if (samsungSerialNumbers.size() > 0) {
33087 amit.gupta 888
                List<AgeingSummaryModel> billedImeiModels = warehouseInventoryItemRepository.findStockAgeingByFofoIdSerialNumbers(retailerId, samsungSerialNumbers);
33606 ranu 889
                List<String> agedSerialNumbers = billedImeiModels.stream().filter(x -> x.isAgedAbove(365)).map(x -> x.getSerialNumber()).collect(Collectors.toList());
890
                if (agedSerialNumbers.size() > 0) {
891
                    List<String> samsungExceptionsSerialNumbers = samsungExceptionRepository.selectAllBySerialNumber(agedSerialNumbers).stream().map(x -> x.getSerialNumber()).collect(Collectors.toList());
892
                    agedSerialNumbers.removeAll(samsungExceptionsSerialNumbers);
893
                }
33432 amit.gupta 894
                inventoryItems = inventoryItems.stream().filter(x -> !agedSerialNumbers.contains(x.getSerialNumber())).collect(Collectors.toSet());
33087 amit.gupta 895
            }
896
 
31903 amit.gupta 897
            if (inventoryItems.size() == 0) return 0;
29927 amit.gupta 898
 
36397 amit 899
            // Resolve GRN billing date per inventory item: InventoryItem.purchaseId -> Purchase.purchaseReference -> Order.billingTimestamp.
900
            // Scheme resolution for OUT must use each item's own GRN billing date, not fofoOrder.createTimestamp.
901
            Set<Integer> purchaseIds = inventoryItems.stream().map(InventoryItem::getPurchaseId).collect(Collectors.toSet());
902
            Map<Integer, LocalDateTime> purchaseBillingDateMap = new HashMap<>();
903
            for (Integer purchaseId : purchaseIds) {
904
                purchaseBillingDateMap.put(purchaseId, purchaseService.getBillingDateOfPurchase(purchaseId));
905
            }
23444 amit.gupta 906
 
36397 amit 907
            // Group items by (GRN billingDate -> catalogId -> items). Items with identical GRN billing date share a scheme lookup.
908
            Map<LocalDateTime, Map<Integer, List<InventoryItem>>> billingDateCatalogItemsMap = inventoryItems.stream()
909
                    .collect(Collectors.groupingBy(
910
                            x -> purchaseBillingDateMap.get(x.getPurchaseId()),
911
                            Collectors.groupingBy(x -> x.getItem().getCatalogItemId())));
31903 amit.gupta 912
 
35493 amit 913
            // N+1 fix: Batch fetch all SchemeInOut records for all inventoryItems at once
914
            Set<Integer> allInventoryItemIds = inventoryItems.stream().map(InventoryItem::getId).collect(Collectors.toSet());
915
            List<SchemeInOut> allSchemeInOuts = schemeInOutRepository.selectByInventoryItemIds(allInventoryItemIds);
916
            Map<Integer, List<SchemeInOut>> schemeInOutByInventoryItemId = allSchemeInOuts.stream()
917
                    .collect(Collectors.groupingBy(SchemeInOut::getInventoryItemId));
34317 amit.gupta 918
 
31410 amit.gupta 919
            int count = 0;
36397 amit 920
            for (Map.Entry<LocalDateTime, Map<Integer, List<InventoryItem>>> billingDateEntry : billingDateCatalogItemsMap.entrySet()) {
921
                LocalDateTime grnBillingDate = billingDateEntry.getKey();
922
                Map<Integer, List<InventoryItem>> catalogInventoryItemMap = billingDateEntry.getValue();
923
                Map<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSchemeSummaryMap = tagListingRepository.getModelSchemesByCatalogIdsAndType(retailerId,
924
                        partnerType, new ArrayList<>(catalogInventoryItemMap.keySet()), grnBillingDate);
34317 amit.gupta 925
 
36397 amit 926
                LOGGER.info("catalogSchemeSummaryMap (grnBillingDate={}) - {}", grnBillingDate, catalogSchemeSummaryMap);
23444 amit.gupta 927
 
36397 amit 928
                for (Map.Entry<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSummaryModelListEntry : catalogSchemeSummaryMap.entrySet()) {
929
                    CatalogSummaryModel catalogSummaryModel = catalogSummaryModelListEntry.getKey();
930
                    List<SchemeSummaryModel> schemeSummaryModels = catalogSummaryModelListEntry.getValue().stream().filter(x -> x != null).collect(Collectors.toList());
35394 amit 931
 
36397 amit 932
                    schemeSummaryModels.stream().filter(x -> x.getSchemeType().getTransactionType().equals(StockTransactionType.OUT)).forEach(x -> x.setProcess(true));
933
                    if (schemeSummaryModels.stream().filter(x -> x.isProcess()).count() == 0) continue;
934
 
935
                    // Create map once per catalog instead of per inventoryItem
936
                    Map<Integer, SchemeSummaryModel> schemeSummaryModelMap = schemeSummaryModels.stream().collect(Collectors.toMap(x -> x.getSchemeId(), x -> x));
937
 
938
                    List<InventoryItem> modelInventoryItems = catalogInventoryItemMap.get(catalogSummaryModel.getCatalogId());
939
                    for (InventoryItem inventoryItem : modelInventoryItems) {
35493 amit 940
                    // N+1 fix: Use pre-fetched schemeInOut map instead of querying per inventoryItem
941
                    List<SchemeInOut> sios = schemeInOutByInventoryItemId.getOrDefault(inventoryItem.getId(), Collections.emptyList());
34740 amit.gupta 942
 
34705 amit.gupta 943
                    List<Integer> creditedSchemeIds = sios.stream()
944
                            .filter(x -> x.getStatus().equals(SchemePayoutStatus.CREDITED))
945
                            .map(x -> x.getSchemeId()).collect(Collectors.toList());
34698 amit.gupta 946
 
34742 amit.gupta 947
                    double sioRejectedValue = 0;
34740 amit.gupta 948
                    if (creditedSchemeIds.size() > 0) {
34778 amit.gupta 949
                        //Also ignore special support
34802 amit.gupta 950
                        //This code block needs to be deleted
34803 amit.gupta 951
                        List<Integer> schemeIdsToReject = null;
34802 amit.gupta 952
                        if (Arrays.asList(1025588, 1025589).contains(inventoryItem.getItem().getCatalogItemId())) {
34803 amit.gupta 953
                            schemeIdsToReject = schemeRepository.selectBySchemeIds(creditedSchemeIds).stream()
34802 amit.gupta 954
                                    .filter(x -> !SchemeType.SPECIAL_SUPPORT.equals(x.getType()))
955
                                    .map(x -> x.getId()).collect(Collectors.toList());
34803 amit.gupta 956
                            //List<Integer> schemeIdsToReject = schemeRepository.selectBySchemeIds(creditedSchemeIds).stream()
957
                            //This code block needs to be deleted
958
                        } else {
959
                            schemeIdsToReject = schemeRepository.selectBySchemeIds(creditedSchemeIds).stream()
960
                                    .filter(x -> SchemeType.OUT_TYPES.contains(x.getType()) && !SchemeType.SPECIAL_SUPPORT.equals(x.getType()))
961
                                    .map(x -> x.getId()).collect(Collectors.toList());
962
                            //Reject invalid scheme payouts due to upgrade in Category or any change in schemes historically
963
                            //Lets not touchs
34802 amit.gupta 964
                        }
34740 amit.gupta 965
                        for (SchemeInOut sio : sios) {
966
                            if (schemeIdsToReject.contains(sio.getSchemeId()) && !schemeSummaryModelMap.containsKey(sio.getSchemeId())) {
35394 amit 967
                                // Removed unused schemeRepository.selectById call
34778 amit.gupta 968
                                sio.setStatusDescription("Rolledback due to Category upgrade/invalid scheme");
969
                                sio.setStatus(SchemePayoutStatus.REJECTED);
970
                                sio.setRolledBackTimestamp(LocalDateTime.now());
971
                                sioRejectedValue += sio.getAmount();
34740 amit.gupta 972
                            }
34705 amit.gupta 973
                        }
34698 amit.gupta 974
                    }
32165 amit.gupta 975
                    float inventoryItemCashback = this.createSchemeInOut(schemeSummaryModels, inventoryItem);
34742 amit.gupta 976
                    if (inventoryItemCashback > 0 || sioRejectedValue > 0) {
31903 amit.gupta 977
                        count++;
34742 amit.gupta 978
                        totalCashback += inventoryItemCashback - sioRejectedValue;
31903 amit.gupta 979
                    }
31410 amit.gupta 980
                }
36397 amit 981
                }
31410 amit.gupta 982
            }
31903 amit.gupta 983
 
31410 amit.gupta 984
            if (count > 0) {
985
                walletService.addAmountToWallet(
986
                        retailerId, fofoOrderId, WalletReferenceType.SCHEME_OUT, "Sales margin for invoice number "
987
                                + fofoOrder.getInvoiceNumber() + ". Total " + count + " pc(s)",
988
                        totalCashback, fofoOrder.getCreateTimestamp());
989
                fofoOrder.setCashback(totalCashback + fofoOrder.getCashback());
990
            }
991
        }
34746 amit.gupta 992
        return totalCashback;
31410 amit.gupta 993
    }
22653 ashik.ali 994
 
31410 amit.gupta 995
    @Override
996
    //Tax rate has been passed to 0 to ensure no tax deduction
997
    public float getSpecialSupportAmount(float supportAmount, PartnerType partnerType, LocalDate onDate,
998
                                         int catalogId) throws ProfitMandiBusinessException {
999
        //int itemId = itemRepository.selectAllByCatalogItemId(catalogId).stream().findAny().get().getId();
1000
        //float totalTaxRate = stateGstRateRepository.getTotalTaxRate(itemId);
1001
        return this.getSpecialSupportAmount(supportAmount, partnerType, onDate, catalogId, 0);
1002
    }
23444 amit.gupta 1003
 
31410 amit.gupta 1004
    @Override
1005
    public float getSpecialSupportAmount(float supportAmount, PartnerType partnerType, LocalDate onDate,
1006
                                         int catalogId, float taxRate) throws ProfitMandiBusinessException {
1007
        float totalMargin = this.selectPercentageScheme(partnerType, onDate, catalogId, false, 0, 0).stream().collect(Collectors.summingDouble(x -> x.getAmount())).floatValue();
1008
        float amountToCredit = supportAmount * (1 - (totalMargin / (100 + taxRate)));
1009
        return amountToCredit;
1010
    }
23444 amit.gupta 1011
 
31410 amit.gupta 1012
    @Override
33614 amit.gupta 1013
    public void rollbackSchemes(List<Integer> inventoryItemIds, String rollbackReason)
31410 amit.gupta 1014
            throws Exception {
33614 amit.gupta 1015
        List<InventoryItem> inventoryItems = inventoryItemRepository.selectAllByIds(inventoryItemIds);
1016
        Map<Integer, InventoryItem> inventoryItemMap = inventoryItems.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
1017
        Map<Integer, Integer> purchasePartnerMap = inventoryItems.stream().collect(Collectors.toMap(x -> x.getPurchaseId(), x -> x.getFofoId(), (u, v) -> u));
1018
 
1019
        LOGGER.info("inventoryItemIds - {}", inventoryItemIds);
1020
        List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(new HashSet<>(inventoryItemIds));
1021
        List<Integer> schemeIds = schemeInOuts.stream().map(x -> x.getSchemeId()).distinct().collect(Collectors.toList());
34139 amit.gupta 1022
        if (schemeIds.size() == 0) return;
33614 amit.gupta 1023
        List<Scheme> schemes = schemeRepository.selectBySchemeIds(schemeIds);
1024
        Map<Integer, Float> inSchemesMap = new HashMap<>();
1025
        Map<Integer, Float> outSchemesMap = new HashMap<>();
1026
        Map<Integer, Scheme> schemesMap = schemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
1027
        for (SchemeInOut schemeInOut : schemeInOuts) {
1028
            Map<Integer, Float> schemePayoutMap;
1029
            int inventoryItemId = schemeInOut.getInventoryItemId();
31410 amit.gupta 1030
            if (schemeInOut.getRolledBackTimestamp() == null) {
1031
                schemeInOut.setRolledBackTimestamp(LocalDateTime.now());
1032
                if (schemeInOut.getStatus() == null || schemeInOut.getStatus().equals(SchemePayoutStatus.CREDITED)) {
33614 amit.gupta 1033
                    Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
34317 amit.gupta 1034
                    if (scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
33614 amit.gupta 1035
                        schemePayoutMap = inSchemesMap;
1036
                    } else {
1037
                        schemePayoutMap = outSchemesMap;
1038
                    }
1039
                    if (!schemePayoutMap.containsKey(inventoryItemId)) {
1040
                        schemePayoutMap.put(inventoryItemId, 0f);
1041
                    }
34139 amit.gupta 1042
                    schemePayoutMap.put(inventoryItemId, inSchemesMap.get(inventoryItemId) == null ? 0 : inSchemesMap.get(inventoryItemId) + schemeInOut.getAmount());
31410 amit.gupta 1043
                }
1044
                schemeInOut.setStatus(SchemePayoutStatus.REJECTED);
1045
                schemeInOut.setStatusDescription(rollbackReason);
1046
            }
1047
        }
33614 amit.gupta 1048
        Map<Integer, Double> purchaseRollbackAmountMap = inSchemesMap.entrySet().stream().collect(Collectors.groupingBy(x -> inventoryItemMap.get(x.getKey()).getPurchaseId(), Collectors.summingDouble(x -> x.getValue())));
1049
 
1050
        for (Map.Entry<Integer, Double> purchaseRollbackAmountEntry : purchaseRollbackAmountMap.entrySet()) {
1051
            int purchaseId = purchaseRollbackAmountEntry.getKey();
1052
            Double amountToRollback = purchaseRollbackAmountEntry.getValue();
1053
            if (amountToRollback != null && amountToRollback > 0) {
1054
                walletService.rollbackAmountFromWallet(purchasePartnerMap.get(purchaseId), amountToRollback.floatValue(), purchaseId,
1055
                        WalletReferenceType.SCHEME_IN, rollbackReason, LocalDateTime.now());
1056
            }
31410 amit.gupta 1057
        }
1058
    }
23444 amit.gupta 1059
 
31410 amit.gupta 1060
    @Override
36103 amit 1061
    public void restoreSchemes(List<Integer> inventoryItemIds, String debitNoteNumber) throws Exception {
1062
        String rollbackDescription = "Schemes rolled back for DebitNote #" + debitNoteNumber;
1063
 
1064
        List<InventoryItem> inventoryItems = inventoryItemRepository.selectAllByIds(inventoryItemIds);
1065
        Map<Integer, InventoryItem> inventoryItemMap = inventoryItems.stream()
1066
                .collect(Collectors.toMap(InventoryItem::getId, x -> x));
1067
        Map<Integer, Integer> purchasePartnerMap = inventoryItems.stream()
1068
                .collect(Collectors.toMap(InventoryItem::getPurchaseId, InventoryItem::getFofoId, (u, v) -> u));
1069
 
1070
        List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(new HashSet<>(inventoryItemIds));
1071
        List<Integer> schemeIds = schemeInOuts.stream().map(SchemeInOut::getSchemeId).distinct()
1072
                .collect(Collectors.toList());
1073
        if (schemeIds.isEmpty()) return;
1074
 
1075
        List<Scheme> schemes = schemeRepository.selectBySchemeIds(schemeIds);
1076
        Map<Integer, Scheme> schemesMap = schemes.stream().collect(Collectors.toMap(Scheme::getId, x -> x));
1077
        Map<Integer, Float> inSchemesRestoreMap = new HashMap<>();
1078
 
1079
        for (SchemeInOut schemeInOut : schemeInOuts) {
1080
            if (schemeInOut.getStatus() != null
1081
                    && schemeInOut.getStatus().equals(SchemePayoutStatus.REJECTED)
1082
                    && rollbackDescription.equals(schemeInOut.getStatusDescription())) {
1083
 
1084
                Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
1085
                if (scheme != null && scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
1086
                    int inventoryItemId = schemeInOut.getInventoryItemId();
1087
                    inSchemesRestoreMap.merge(inventoryItemId, schemeInOut.getAmount(), Float::sum);
1088
                }
1089
 
1090
                schemeInOut.setStatus(SchemePayoutStatus.CREDITED);
1091
                schemeInOut.setStatusDescription("Restored: DN rejection reversal #" + debitNoteNumber);
1092
                schemeInOut.setRolledBackTimestamp(null);
1093
            }
1094
        }
1095
 
1096
        Map<Integer, Double> purchaseRestoreAmountMap = inSchemesRestoreMap.entrySet().stream()
1097
                .collect(Collectors.groupingBy(
1098
                        x -> inventoryItemMap.get(x.getKey()).getPurchaseId(),
1099
                        Collectors.summingDouble(Map.Entry::getValue)));
1100
 
1101
        String restoreReason = "Scheme restored: DN rejection reversal #" + debitNoteNumber;
1102
        for (Map.Entry<Integer, Double> entry : purchaseRestoreAmountMap.entrySet()) {
1103
            int purchaseId = entry.getKey();
1104
            Double amountToRestore = entry.getValue();
1105
            if (amountToRestore != null && amountToRestore > 0) {
1106
                walletService.addAmountToWallet(purchasePartnerMap.get(purchaseId), purchaseId,
1107
                        WalletReferenceType.SCHEME_IN, restoreReason,
1108
                        amountToRestore.floatValue(), LocalDateTime.now());
1109
            }
1110
        }
1111
    }
1112
 
1113
    @Override
31410 amit.gupta 1114
    public Map<String, Object> getSchemes(Set<Integer> roleIds, int offset, int limit)
1115
            throws ProfitMandiBusinessException {
1116
        Map<String, Object> map = new HashMap<>();
1117
        List<Scheme> schemes = null;
1118
        long size = 0;
1119
        if (roleManager.isAdmin(roleIds)) {
1120
            schemes = schemeRepository.selectAll(offset, limit);
1121
            size = schemeRepository.selectAllCount();
1122
        } else {
1123
            schemes = schemeRepository.selectActiveAll(offset, limit);
1124
            size = schemeRepository.selectAllActiveCount();
1125
        }
1126
        map.put("schemes", schemes);
1127
        map.put("start", offset + 1);
1128
        map.put("size", size);
1129
        if (schemes.size() < limit) {
1130
            map.put("end", offset + schemes.size());
1131
        } else {
1132
            map.put("end", offset + limit);
1133
        }
1134
        return map;
1135
    }
26722 amit.gupta 1136
 
31410 amit.gupta 1137
    @Override
1138
    public List<Scheme> getPaginatedSchemes(Set<Integer> roleIds, int offset, int limit)
1139
            throws ProfitMandiBusinessException {
1140
        LOGGER.info("requested offset=[{}], limit = [{}]", offset, limit);
1141
        List<Scheme> schemes = null;
1142
        if (roleManager.isAdmin(roleIds)) {
1143
            schemes = schemeRepository.selectAll(offset, limit);
1144
        } else {
1145
            schemes = schemeRepository.selectActiveAll(offset, limit);
1146
        }
1147
        return schemes;
1148
    }
27898 amit.gupta 1149
 
31410 amit.gupta 1150
    @Override
1151
    // This is being called to reverse schemes while processing price Drop
1152
    public void reverseSchemes(List<InventoryItem> inventoryItems, int priceDropId, String reversalReason)
1153
            throws ProfitMandiBusinessException {
1154
        PriceDrop priceDrop = priceDropRepository.selectById(priceDropId);
1155
        Map<Integer, List<InventoryItem>> purchaseInventoryListMap = inventoryItems.stream()
1156
                .collect(Collectors.groupingBy(InventoryItem::getPurchaseId, Collectors.toList()));
29927 amit.gupta 1157
 
31410 amit.gupta 1158
        for (Map.Entry<Integer, List<InventoryItem>> purchaseEntry : purchaseInventoryListMap.entrySet()) {
1159
            float amountToCredit = 0;
1160
            float amountToDebit = 0;
1161
            int purchaseId = purchaseEntry.getKey();
1162
            List<InventoryItem> purchaseInventoryItemList = purchaseEntry.getValue();
30454 amit.gupta 1163
 
31410 amit.gupta 1164
            Map<Integer, InventoryItem> inventoryItemsMap = purchaseInventoryItemList.stream()
1165
                    .collect(Collectors.toMap(x -> x.getId(), x -> x));
29927 amit.gupta 1166
 
31410 amit.gupta 1167
            List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(inventoryItemsMap.keySet());
1168
            LOGGER.info("Scheme InOuts , {}", schemeInOuts);
1169
            if (schemeInOuts.size() == 0) {
1170
                continue;
1171
            }
1172
            List<Integer> schemeIds = schemeInOuts.stream().map(x -> x.getSchemeId()).collect(Collectors.toList());
1173
            Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds, 0, schemeIds.size())
1174
                    .stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
1175
            for (SchemeInOut schemeInOut : schemeInOuts) {
1176
                InventoryItem ii = inventoryItemsMap.get(schemeInOut.getInventoryItemId());
1177
                Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
1178
                if (scheme.getAmountType().equals(AmountType.FIXED)) {
1179
                    continue;
1180
                }
34317 amit.gupta 1181
                if (scheme.getType().getTransactionType().equals(StockTransactionType.IN) && schemeInOut.getRolledBackTimestamp() == null) {
33432 amit.gupta 1182
                    schemeInOut.setRolledBackTimestamp(LocalDateTime.now());
1183
                    schemeInOut.setStatus(SchemePayoutStatus.REJECTED);
1184
                    schemeInOut.setStatusDescription("Margin reversed due to price drop");
1185
                    // IF not credited then dont consider any credit/debit for that sio entry
1186
                    if (schemeInOut.getCreditTimestamp() != null) {
1187
                        amountToDebit += schemeInOut.getAmount();
31410 amit.gupta 1188
                    }
1189
                }
1190
            }
1191
            int fofoId = inventoryItems.get(0).getFofoId();
1192
            if (amountToDebit > 0) {
1193
                walletService.addAmountToWallet(fofoId, purchaseId, WalletReferenceType.SCHEME_IN,
1194
                        MessageFormat.format(reversalReason, purchaseInventoryItemList.size()), -amountToDebit,
1195
                        priceDrop.getAffectedOn());
1196
            }
1197
        }
1198
    }
29231 amit.gupta 1199
 
31609 amit.gupta 1200
    @Autowired
32060 amit.gupta 1201
    UserWalletRepository userWalletRepository;
1202
    @Autowired
31609 amit.gupta 1203
    UserWalletHistoryRepository userWalletHistoryRepository;
1204
 
31410 amit.gupta 1205
    @Override
1206
    // Always being called from cancel order/bad return means no SCHEME IN is considered
1207
    public void reverseSchemes(List<InventoryItem> inventoryItems, int reversalReference, String reversalReason,
1208
                               List<SchemeType> schemeTypes) throws ProfitMandiBusinessException {
1209
        Map<Integer, InventoryItem> inventoryItemsMap = inventoryItems.stream()
1210
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1211
        LOGGER.info("inventoryItems" + inventoryItems);
23444 amit.gupta 1212
 
31410 amit.gupta 1213
        Map<SchemeType, SchemeInOut> schemeTypeMap = new HashMap<>();
30454 amit.gupta 1214
 
31410 amit.gupta 1215
        List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(inventoryItemsMap.keySet());
32060 amit.gupta 1216
        List<SchemeInOut> rolledBacks = schemeInOuts.stream().filter(x -> x.getStatusDescription().equals(reversalReason)).collect(Collectors.toList());
31410 amit.gupta 1217
        float amountToRollback = 0;
23444 amit.gupta 1218
 
31410 amit.gupta 1219
        if (!schemeInOuts.isEmpty()) {
1220
            List<Integer> schemeIds = schemeInOuts.stream().map(x -> x.getSchemeId()).collect(Collectors.toList());
1221
            LOGGER.info("schemeIds" + schemeIds);
29231 amit.gupta 1222
 
31410 amit.gupta 1223
            Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds, 0, schemeIds.size())
1224
                    .stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
32060 amit.gupta 1225
            if (rolledBacks.size() > 0) {
1226
                for (SchemeInOut schemeInOut : rolledBacks) {
1227
                    Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
1228
                    if (schemeTypes.contains(scheme.getType())) {
1229
                        schemeTypeMap.put(scheme.getType(), schemeInOut);
1230
                        if (schemeInOut.getCreditTimestamp() != null) {
31410 amit.gupta 1231
                            amountToRollback += schemeInOut.getAmount();
1232
                        }
1233
                    }
1234
                }
32060 amit.gupta 1235
            } else {
1236
                for (SchemeInOut schemeInOut : schemeInOuts) {
1237
                    Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
1238
                    if (schemeTypes.contains(scheme.getType())) {
1239
                        if (schemeInOut.getRolledBackTimestamp() == null) {
1240
                            schemeInOut.setRolledBackTimestamp(LocalDateTime.now());
1241
                            if (schemeInOut.getStatus().equals(SchemePayoutStatus.CREDITED)) {
1242
                                amountToRollback += schemeInOut.getAmount();
1243
                            }
1244
                            schemeInOut.setStatus(SchemePayoutStatus.REJECTED);
1245
                            schemeInOut.setStatusDescription(reversalReason);
1246
                        }
1247
                    }
1248
                }
31410 amit.gupta 1249
            }
1250
        }
32060 amit.gupta 1251
 
31410 amit.gupta 1252
        int fofoId = inventoryItems.get(0).getFofoId();
34486 amit.gupta 1253
        WalletReferenceType walletReferenceType = schemeTypes.containsAll(SchemeType.OUT_SCHEME_TYPES) ? WalletReferenceType.SCHEME_OUT
31609 amit.gupta 1254
                : schemeTypes.contains(SchemeType.SPECIAL_SUPPORT) ? WalletReferenceType.SPECIAL_SUPPORT
34486 amit.gupta 1255
                : schemeTypes.contains(SchemeType.INVESTMENT) ? WalletReferenceType.SCHEME_OUT : null;
32060 amit.gupta 1256
        List<UserWalletHistory> userWalletHistoryList = null;
32058 amit.gupta 1257
        if (amountToRollback > 0 && walletReferenceType != null) {
1258
            // Mark appropriate reference of rollback investment margin
32060 amit.gupta 1259
            if (schemeTypes.contains(SchemeType.INVESTMENT) && schemeTypeMap.containsKey(SchemeType.INVESTMENT)) {
1260
                LocalDateTime creditTime = schemeTypeMap.get(SchemeType.INVESTMENT).getCreditTimestamp();
1261
                if (creditTime == null) return;
1262
                int investmentReversalReference = Integer.parseInt(FormattingUtils.getYearMonth(creditTime.minusMonths(1)));
1263
                UserWallet userWallet = userWalletRepository.selectByRetailerId(fofoId);
1264
                userWalletHistoryList = userWalletHistoryRepository.selectAllByreferenceIdandreferenceType(userWallet.getUserId(), investmentReversalReference, WalletReferenceType.INVESTMENT_PAYOUT);
1265
                if (userWalletHistoryList.size() > 0) {
1266
                    walletReferenceType = WalletReferenceType.INVESTMENT_PAYOUT;
1267
                } else {
1268
                    userWalletHistoryList = null;
32058 amit.gupta 1269
                }
32060 amit.gupta 1270
            }
1271
            if (userWalletHistoryList == null) {
1272
                userWalletHistoryList = userWalletHistoryRepository.selectAllByreferenceIdandreferenceType(reversalReference, walletReferenceType);
1273
            }
1274
            if (userWalletHistoryList.size() > 0) {
1275
                int maxDeductible = userWalletHistoryList.stream().collect(Collectors.summingInt(x -> x.getAmount()));
1276
                if (maxDeductible > 0) {
32165 amit.gupta 1277
                    LOGGER.info("----------> maxDeductible {}, amountToRollback {}, reversalReference {}, walletReferenceType {} ", maxDeductible, amountToRollback, reversalReference, walletReferenceType);
32060 amit.gupta 1278
                    walletService.rollbackAmountFromWallet(fofoId, Math.min(maxDeductible, amountToRollback), reversalReference, walletReferenceType,
1279
                            reversalReason, LocalDateTime.now());
1280
                }
31609 amit.gupta 1281
 
31903 amit.gupta 1282
            }
32058 amit.gupta 1283
        }
31410 amit.gupta 1284
    }
29231 amit.gupta 1285
 
32060 amit.gupta 1286
 
31410 amit.gupta 1287
    @Override
33614 amit.gupta 1288
    public Map<Integer, Float> getCatalogSchemeCashBack(int fofoId, List<Integer> catalogIds) throws
1289
            ProfitMandiBusinessException {
32972 amit.gupta 1290
        PartnerType partnerType = partnerTypeChangeService.getTypeOnDate(fofoId, LocalDate.now());
1291
        Map<CatalogSummaryModel, List<SchemeSummaryModel>> catalogModelMap = tagListingRepository.getModelSchemesByCatalogIdsAndType(fofoId, partnerType, catalogIds, LocalDate.now().atStartOfDay());
1292
 
1293
        Map<Integer, Float> catalogCashbackMap = new HashMap<>();
1294
        for (Map.Entry<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSummaryModelListEntry : catalogModelMap.entrySet()) {
1295
            int catalogItemId = catalogSummaryModelListEntry.getKey().getCatalogId();
1296
            List<SchemeSummaryModel> schemeSummaryModels = catalogSummaryModelListEntry.getValue();
1297
 
1298
            float totalCashback = schemeSummaryModels.stream()
34317 amit.gupta 1299
                    .filter(x -> Arrays.asList(SchemeType.SPECIAL_SUPPORT, SchemeType.SELLOUT).contains(x.getSchemeType())
32972 amit.gupta 1300
                            && x.getAmountType().equals(AmountType.FIXED))
1301
                    .collect(Collectors.summingDouble(x -> x.getAmount())).floatValue();
1302
            catalogCashbackMap.put(catalogItemId, totalCashback);
31410 amit.gupta 1303
        }
32972 amit.gupta 1304
        return catalogCashbackMap;
31410 amit.gupta 1305
    }
30572 amit.gupta 1306
 
31410 amit.gupta 1307
    @Override
31903 amit.gupta 1308
    public List<Scheme> selectSchemeByPartnerTypeFofoId(PartnerType partnerType, LocalDate onDate, int catalogId,
1309
                                                        int fofoId, int offset, int limit) throws ProfitMandiBusinessException {
31410 amit.gupta 1310
        Session session = sessionFactory.getCurrentSession();
1311
        final TypedQuery<Scheme> typedQuery = session.createNamedQuery(
1312
                "Scheme.selectSchemeByModelsPartnerTypeFofoId", Scheme.class);
1313
        typedQuery.setParameter("catalogIds", Arrays.asList(catalogId));
1314
        typedQuery.setParameter("fofoIds", Arrays.asList(fofoId, 0));
1315
        typedQuery.setParameter("onDate", onDate.atStartOfDay());
1316
        typedQuery.setParameter("partnerTypes", Arrays.asList(partnerType, partnerType.ALL));
1317
        typedQuery.setFirstResult(offset);
1318
        if (limit != 0) {
1319
            typedQuery.setMaxResults(limit);
1320
        }
1321
        return typedQuery.getResultList();
1322
    }
30463 amit.gupta 1323
 
33432 amit.gupta 1324
    @Override
1325
    public void processSchemeIn(List<InventoryItem> inventoryItems) throws ProfitMandiBusinessException {
1326
        Map<Integer, List<InventoryItem>> purchaseIdInventoryItemsMap = inventoryItems.stream().collect(Collectors.groupingBy(x -> x.getPurchaseId()));
1327
        for (Map.Entry<Integer, List<InventoryItem>> purchaseIdInventoryItemEntry : purchaseIdInventoryItemsMap.entrySet()) {
34317 amit.gupta 1328
            int retailerId = purchaseIdInventoryItemEntry.getValue().get(0).getFofoId();
1329
            this.processSchemeIn(purchaseIdInventoryItemEntry.getKey(), retailerId);
33432 amit.gupta 1330
        }
1331
    }
1332
 
32309 amit.gupta 1333
    @Autowired
1334
    FofoStoreRepository fofoStoreRepository;
1335
 
1336
 
31410 amit.gupta 1337
    @Override
32309 amit.gupta 1338
    @Cacheable(value = "staticscheme", cacheManager = "oneDayCacheManager")
1339
    public Scheme getStaticScheme(int fofoId) throws ProfitMandiBusinessException {
33165 amit.gupta 1340
        FofoStore fofoStore = fofoStoreRepository.selectByRetailerId(fofoId);
32504 amit.gupta 1341
        Scheme scheme = null;
1342
        if (fofoStore.getTarget() > 0) {
1343
            scheme = new Scheme();
35096 amit 1344
            scheme.setName("Super Retailer - Club 4");
1345
            scheme.setStartDateTime(LocalDate.of(2025, 8, 1).atStartOfDay());
35559 amit 1346
            scheme.setEndDateTime(LocalDate.of(2026, 01, 20).atTime(LocalTime.MAX));
32504 amit.gupta 1347
            scheme.setTarget(fofoStore.getTarget());
33548 amit.gupta 1348
 
33704 amit.gupta 1349
            if (scheme.getEndDateTime().plusDays(5).isBefore(LocalDateTime.now())) return null;
32504 amit.gupta 1350
        }
33165 amit.gupta 1351
        return scheme;
32309 amit.gupta 1352
    }
1353
 
1354
    @Override
31410 amit.gupta 1355
    public List<Scheme> selectSchemeByPartnerType(PartnerType partnerType, LocalDate onDate, int catalogId,
1356
                                                  boolean isAdmin, int offset, int limit) throws ProfitMandiBusinessException {
1357
        Session session = sessionFactory.getCurrentSession();
1358
        List<Predicate> andPredicates = new ArrayList<>();
1359
        CriteriaBuilder cb = session.getCriteriaBuilder();
1360
        CriteriaQuery<Scheme> query = cb.createQuery(Scheme.class);
1361
        Root<Scheme> scheme = query.from(Scheme.class);
1362
        if (!partnerType.equals(PartnerType.ALL)) {
1363
            List<PartnerType> pt = new ArrayList<>();
1364
            pt.add(PartnerType.ALL);
1365
            pt.add(partnerType);
1366
            andPredicates.add(cb.in(scheme.get("partnerType")).value(pt));
1367
        }
1368
        cb.desc(cb.isNull(scheme.get("expireTimestamp")));
1369
        if (catalogId > 0) {
30454 amit.gupta 1370
 
31410 amit.gupta 1371
            List<Integer> schemeIds = schemeItemRepository.selectSchemeIdByCatalogId(catalogId);
1372
            LOGGER.info("schemeId" + schemeIds);
1373
            if (schemeIds.isEmpty()) {
1374
                return new ArrayList<>();
1375
            }
1376
            andPredicates.add(cb.in(scheme.get("id")).value(schemeIds));
1377
            if (onDate != null) {
1378
                andPredicates.add(cb.greaterThan(scheme.get("endDateTime"), onDate.atStartOfDay()));
1379
                andPredicates.add(cb.lessThanOrEqualTo(scheme.get("startDateTime"), onDate.atStartOfDay()));
1380
            }
1381
        }
1382
        if (!isAdmin) {
1383
            andPredicates.add(cb.isNotNull(scheme.get("activeTimestamp")));
1384
        }
1385
        query.where(cb.and(andPredicates.toArray(new Predicate[0])));
1386
        query.orderBy(cb.desc(cb.function("isnull", Boolean.class, scheme.get("expireTimestamp"))));
1387
        if (limit == 0) {
1388
            return session.createQuery(query).setFirstResult(offset).getResultList();
1389
        }
1390
        return session.createQuery(query).setFirstResult(offset).setMaxResults(limit).getResultList();
30454 amit.gupta 1391
 
31410 amit.gupta 1392
    }
30454 amit.gupta 1393
 
31410 amit.gupta 1394
    @Override
1395
    public List<Scheme> selectPercentageScheme(PartnerType partnerType, LocalDate onDate, int catalogId,
1396
                                               boolean isAdmin, int offset, int limit) throws ProfitMandiBusinessException {
1397
        List<Scheme> schemes = this.selectSchemeByPartnerType(partnerType, onDate, catalogId, isAdmin, offset, limit);
1398
        return schemes.stream().filter(x -> x.getAmountType().equals(AmountType.PERCENTAGE)).collect(Collectors.toList());
1399
    }
30454 amit.gupta 1400
 
31410 amit.gupta 1401
    @Override
1402
    public void processActivation() throws ProfitMandiBusinessException {
1403
        List<SchemeInOut> pendingPayouts = schemeInOutRepository.selectAllPending();
1404
        List<Integer> schemeIds = new ArrayList<>();
1405
        Set<Integer> inventoryIds = new HashSet<>();
1406
        for (SchemeInOut pendingPayout : pendingPayouts) {
1407
            schemeIds.add(pendingPayout.getSchemeId());
1408
        }
1409
        Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds, 0, 0).stream()
34317 amit.gupta 1410
                .filter(x -> x.getType().equals(SchemeType.SPECIAL_SUPPORT))
31410 amit.gupta 1411
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1412
        pendingPayouts = pendingPayouts.stream().filter(x -> schemesMap.get(x.getSchemeId()) != null)
1413
                .collect(Collectors.toList());
23884 amit.gupta 1414
 
31410 amit.gupta 1415
        for (SchemeInOut pendingPayout : pendingPayouts) {
1416
            inventoryIds.add(pendingPayout.getInventoryItemId());
1417
        }
1418
        Map<Integer, InventoryItem> inventoryItemMap = inventoryItemRepository.selectByIds(inventoryIds).stream()
1419
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1420
        Map<String, InventoryItem> serialNumberMap = inventoryItemMap.values().stream()
1421
                .collect(Collectors.toMap(x -> x.getSerialNumber(), x -> x));
23796 amit.gupta 1422
 
31410 amit.gupta 1423
        List<ActivatedImei> activatedImeis = activatedImeiRepository
1424
                .selectBySerialNumbers(new ArrayList<>(serialNumberMap.keySet())).stream().collect(Collectors.toList());
23508 amit.gupta 1425
 
31410 amit.gupta 1426
        Map<String, ActivatedImei> activatedImeiMap = activatedImeis.stream()
1427
                .collect(Collectors.toMap(x -> x.getSerialNumber().toLowerCase(), x -> x));
1428
        for (SchemeInOut pendingPayout : pendingPayouts) {
31914 amit.gupta 1429
            Scheme scheme = schemesMap.get(pendingPayout.getSchemeId());
31410 amit.gupta 1430
            InventoryItem ii = inventoryItemMap.get(pendingPayout.getInventoryItemId());
1431
            String serialNumber = ii.getSerialNumber().toLowerCase();
1432
            ActivatedImei activatedImei = activatedImeiMap.get(serialNumber);
1433
            if (activatedImei == null) {
1434
                continue;
1435
            }
1436
            if (scheme.isWithinRange(activatedImei.getActivationTimestamp())) {
1437
                int fofoId = ii.getFofoId();
1438
                // Get latest order Id
1439
                int orderId = scanRecordRepository.selectByInventoryItemId(ii.getId()).stream()
1440
                        .filter(x -> x.getOrderId() > 0)
1441
                        .sorted(Comparator.comparing(ScanRecord::getCreateTimestamp).reversed()).findFirst().get()
1442
                        .getOrderId();
1443
                FofoOrder fofoOrder = fofoOrderRepository.selectByOrderId(orderId);
31916 amit.gupta 1444
 
31921 amit.gupta 1445
                InventoryPayoutModel inventoryPayoutModel = priceCircularService.getPayouts(ii);
31916 amit.gupta 1446
                AmountModel amountModel = new AmountModel();
1447
                amountModel.setAmount(scheme.getAmount());
1448
                amountModel.setAmountType(scheme.getAmountType());
31921 amit.gupta 1449
                double amountToRollout = inventoryPayoutModel.getRolloutAmount(amountModel);
32058 amit.gupta 1450
                pendingPayout.setAmount((float) amountToRollout);
34317 amit.gupta 1451
                walletService.addAmountToWallet(fofoId, orderId, WalletReferenceType.SPECIAL_SUPPORT,
1452
                        "Special Support for " + ii.getItem().getItemDescriptionNoColor() + ", Imei - " + serialNumber, (float) amountToRollout,
1453
                        fofoOrder.getCreateTimestamp());
1454
                pendingPayout.setStatusDescription("Special support credited, activated on " + FormattingUtils.formatDate(activatedImei.getActivationTimestamp()));
1455
 
31410 amit.gupta 1456
                pendingPayout.setCreditTimestamp(LocalDateTime.now());
1457
                pendingPayout.setStatus(SchemePayoutStatus.CREDITED);
1458
            } else {
1459
                pendingPayout.setStatus(SchemePayoutStatus.REJECTED);
1460
                pendingPayout.setRolledBackTimestamp(LocalDateTime.now());
1461
                pendingPayout.setStatusDescription(
1462
                        "Rejected, activated on " + FormattingUtils.formatDate(activatedImei.getActivationTimestamp()));
1463
            }
1464
        }
1465
    }
24976 amit.gupta 1466
 
31410 amit.gupta 1467
    @Override
35629 amit 1468
    public void rejectActivatedSchemeIds(List<Integer> schemeIds) throws ProfitMandiBusinessException {
1469
        List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectBySchemeIds(new HashSet<>(schemeIds));
1470
        // Filter to CREDITED records that haven't been rolled back
1471
        List<SchemeInOut> creditedPayouts = schemeInOuts.stream()
1472
                .filter(x -> SchemePayoutStatus.CREDITED.equals(x.getStatus()) && x.getRolledBackTimestamp() == null)
1473
                .collect(Collectors.toList());
1474
        if (creditedPayouts.isEmpty()) return;
1475
 
1476
        Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds).stream()
1477
                .filter(x -> SchemeType.SPECIAL_SUPPORT.equals(x.getType()))
1478
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1479
        creditedPayouts = creditedPayouts.stream().filter(x -> schemesMap.containsKey(x.getSchemeId()))
1480
                .collect(Collectors.toList());
1481
        if (creditedPayouts.isEmpty()) return;
1482
 
1483
        Set<Integer> inventoryIds = creditedPayouts.stream().map(SchemeInOut::getInventoryItemId).collect(Collectors.toSet());
1484
        Map<Integer, InventoryItem> inventoryItemMap = inventoryItemRepository.selectByIds(inventoryIds).stream()
1485
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1486
        List<String> serialNumbers = inventoryItemMap.values().stream().map(InventoryItem::getSerialNumber).collect(Collectors.toList());
1487
        Map<String, ActivatedImei> activatedImeiMap = activatedImeiRepository.selectBySerialNumbers(serialNumbers).stream()
1488
                .collect(Collectors.toMap(x -> x.getSerialNumber().toLowerCase(), x -> x, (a, b) -> a));
1489
 
1490
        for (SchemeInOut payout : creditedPayouts) {
1491
            Scheme scheme = schemesMap.get(payout.getSchemeId());
1492
            InventoryItem ii = inventoryItemMap.get(payout.getInventoryItemId());
1493
            if (ii == null) continue;
1494
            ActivatedImei activatedImei = activatedImeiMap.get(ii.getSerialNumber().toLowerCase());
1495
            if (activatedImei == null || activatedImei.getActivationTimestamp() == null) continue;
1496
 
1497
            if (!scheme.isWithinRange(activatedImei.getActivationTimestamp())) {
1498
                payout.setStatus(SchemePayoutStatus.REJECTED);
1499
                payout.setRolledBackTimestamp(LocalDateTime.now());
1500
                payout.setStatusDescription("Rejected, activation on " + FormattingUtils.formatDate(activatedImei.getActivationTimestamp()) + " outside scheme period");
1501
 
1502
                List<ScanRecord> scanRecords = scanRecordRepository.selectByInventoryItemId(ii.getId());
1503
                int orderId = scanRecords.stream()
1504
                        .filter(x -> x.getOrderId() > 0)
1505
                        .sorted(Comparator.comparing(ScanRecord::getCreateTimestamp).reversed()).findFirst()
1506
                        .map(ScanRecord::getOrderId).orElse(0);
1507
                if (orderId > 0) {
1508
                    walletService.rollbackAmountFromWallet(ii.getFofoId(), payout.getAmount(), orderId,
1509
                            WalletReferenceType.SPECIAL_SUPPORT,
1510
                            "Rejected, activation on " + FormattingUtils.formatDate(activatedImei.getActivationTimestamp()) + " outside scheme period",
1511
                            LocalDateTime.now());
1512
                }
1513
            }
1514
        }
1515
    }
1516
 
1517
    @Override
31410 amit.gupta 1518
    public void processActivatedImeisForSchemes() throws ProfitMandiBusinessException {
1519
        List<SchemesImeisModel> schemesImeisModels = schemeRepository.selectSelectUnpaidSchemes();
1520
        LOGGER.info("Total Size - " + schemesImeisModels.size());
1521
        List<Integer> orderIds = schemesImeisModels.stream().map(x -> x.getOrderId()).collect(Collectors.toList());
1522
        List<FofoOrder> fofoOrders = fofoOrderRepository.selectAllByOrderIds(orderIds);
1523
        Map<Integer, FofoOrder> validOrdersMap = fofoOrders.stream().filter(x -> x.getCancelledTimestamp() == null).collect(Collectors.toMap(x -> x.getId(), x -> x));
1524
        Map<String, List<SchemesImeisModel>> validImeiSchemesModelMap = schemesImeisModels.stream().filter(x -> validOrdersMap.containsKey(x.getOrderId())).collect(Collectors.groupingBy(x -> x.getImei()));
35394 amit 1525
 
1526
        // Batch fetch all inventoryItems to avoid N+1 queries
1527
        Set<Integer> inventoryItemIds = validImeiSchemesModelMap.values().stream()
1528
                .map(list -> list.get(0).getInventoryItemId())
1529
                .collect(Collectors.toSet());
1530
        Map<Integer, InventoryItem> inventoryItemsMap = inventoryItemIds.isEmpty()
1531
                ? Collections.emptyMap()
1532
                : inventoryItemRepository.selectAllByIds(new ArrayList<>(inventoryItemIds)).stream()
1533
                        .collect(Collectors.toMap(InventoryItem::getId, x -> x));
1534
 
31410 amit.gupta 1535
        for (Map.Entry<String, List<SchemesImeisModel>> imeiListEntry : validImeiSchemesModelMap.entrySet()) {
1536
            SchemesImeisModel schemesImeisModel = imeiListEntry.getValue().get(0);
1537
            List<Integer> schemeIds = imeiListEntry.getValue().stream().map(x -> x.getSchemeId()).collect(Collectors.toList());
1538
            LOGGER.info("Serial Number  - {}, Scheme IDs - {}", schemesImeisModel.getImei(), schemeIds);
35394 amit 1539
            InventoryItem inventoryItem = inventoryItemsMap.get(schemesImeisModel.getInventoryItemId());
1540
            if (inventoryItem == null) continue;
31410 amit.gupta 1541
            List<Scheme> schemes = schemeRepository.selectBySchemeIds(schemeIds);
34317 amit.gupta 1542
            List<Scheme> supportSchemes = schemes.stream().filter(x -> Arrays.asList(SchemeType.SPECIAL_SUPPORT).contains(x.getType())).collect(Collectors.toList());
31410 amit.gupta 1543
            if (supportSchemes.size() > 0) {
32165 amit.gupta 1544
                for (Scheme scheme : supportSchemes) {
1545
                    List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByScheme(scheme.getId(), inventoryItem.getId());
32972 amit.gupta 1546
                    if (schemeInOuts.stream().filter(x -> Arrays.asList(SchemePayoutStatus.CREDITED, SchemePayoutStatus.PENDING).contains(x.getStatus())).count() > 0) {
32165 amit.gupta 1547
                        continue;
1548
                    }
1549
                    SchemeInOut sio = new SchemeInOut();
1550
                    sio.setAmount(0);
1551
                    sio.setInventoryItemId(inventoryItem.getId());
1552
                    sio.setSchemeId(scheme.getId());
1553
                    sio.setStatusDescription("Activation pending for IMEI#" + inventoryItem.getSerialNumber());
1554
                    sio.setStatus(SchemePayoutStatus.PENDING);
1555
                    schemeInOutRepository.persist(sio);
1556
                }
31410 amit.gupta 1557
            }
1558
        }
1559
    }
36397 amit 1560
 
1561
    @Override
1562
    public void updateSchemeItemWindow(long schemeItemId, LocalDateTime startDate, LocalDateTime endDate, int updatedBy)
1563
            throws ProfitMandiBusinessException {
1564
        SchemeItem item = schemeItemRepository.selectById(schemeItemId);
1565
        if (item == null) {
1566
            throw new ProfitMandiBusinessException("schemeItemId", String.valueOf(schemeItemId), "Scheme item not found");
1567
        }
1568
 
1569
        Scheme scheme = schemeRepository.selectById(item.getSchemeId());
1570
        if (scheme == null) {
1571
            throw new ProfitMandiBusinessException("schemeId", String.valueOf(item.getSchemeId()), "Scheme not found");
1572
        }
1573
 
1574
        if (startDate.isAfter(endDate)) {
1575
            throw new ProfitMandiBusinessException("startDate, endDate",
1576
                    FormattingUtils.formatDate(startDate) + ", " + FormattingUtils.formatDate(endDate),
1577
                    "Start date must be before end date");
1578
        }
1579
 
1580
        // Containment: item window must be within scheme window
1581
        if (startDate.isBefore(scheme.getStartDateTime()) || endDate.isAfter(scheme.getEndDateTime())) {
1582
            throw new ProfitMandiBusinessException("startDate, endDate",
1583
                    FormattingUtils.formatDate(startDate) + ", " + FormattingUtils.formatDate(endDate),
1584
                    "Item dates must be within scheme window ["
1585
                            + FormattingUtils.formatDate(scheme.getStartDateTime()) + " - "
1586
                            + FormattingUtils.formatDate(scheme.getEndDateTime()) + "]");
1587
        }
1588
 
1589
        // Non-overlap check within same (catalogId, schemeId)
1590
        if (schemeItemRepository.existsOverlapping(item.getCatalogId(), item.getSchemeId(), startDate, endDate, item.getId())) {
1591
            throw new ProfitMandiBusinessException("startDate, endDate",
1592
                    FormattingUtils.formatDate(startDate) + ", " + FormattingUtils.formatDate(endDate),
1593
                    "Date range overlaps with existing entry for this model in this scheme");
1594
        }
1595
 
1596
        item.setStartDate(startDate);
1597
        item.setEndDate(endDate);
1598
        item.setUpdatedBy(updatedBy);
1599
        item.setUpdatedOn(LocalDateTime.now());
1600
        schemeItemRepository.persist(item);
1601
    }
1602
 
1603
    @Override
1604
    public SchemeItem addSchemeItemWithDates(int schemeId, int catalogId, LocalDateTime startDate, LocalDateTime endDate, int createdBy)
1605
            throws ProfitMandiBusinessException {
1606
        Scheme scheme = schemeRepository.selectById(schemeId);
1607
        if (scheme == null) {
1608
            throw new ProfitMandiBusinessException("schemeId", String.valueOf(schemeId), "Scheme not found");
1609
        }
1610
 
1611
        if (startDate.isAfter(endDate)) {
1612
            throw new ProfitMandiBusinessException("startDate, endDate",
1613
                    FormattingUtils.formatDate(startDate) + ", " + FormattingUtils.formatDate(endDate),
1614
                    "Start date must be before end date");
1615
        }
1616
 
1617
        // Containment
1618
        if (startDate.isBefore(scheme.getStartDateTime()) || endDate.isAfter(scheme.getEndDateTime())) {
1619
            throw new ProfitMandiBusinessException("startDate, endDate",
1620
                    FormattingUtils.formatDate(startDate) + ", " + FormattingUtils.formatDate(endDate),
1621
                    "Item dates must be within scheme window ["
1622
                            + FormattingUtils.formatDate(scheme.getStartDateTime()) + " - "
1623
                            + FormattingUtils.formatDate(scheme.getEndDateTime()) + "]");
1624
        }
1625
 
1626
        // Non-overlap
1627
        if (schemeItemRepository.existsOverlapping(catalogId, schemeId, startDate, endDate, 0)) {
1628
            throw new ProfitMandiBusinessException("startDate, endDate",
1629
                    FormattingUtils.formatDate(startDate) + ", " + FormattingUtils.formatDate(endDate),
1630
                    "Date range overlaps with existing entry for this model in this scheme");
1631
        }
1632
 
1633
        SchemeItem item = new SchemeItem();
1634
        item.setSchemeId(schemeId);
1635
        item.setCatalogId(catalogId);
1636
        item.setStartDate(startDate);
1637
        item.setEndDate(endDate);
1638
        item.setCreateTimestamp(LocalDateTime.now());
1639
        item.setUpdatedBy(createdBy);
1640
        item.setUpdatedOn(LocalDateTime.now());
1641
        schemeItemRepository.persist(item);
1642
        return item;
1643
    }
1644
 
1645
    @Override
1646
    public List<SchemeItem> clampSchemeItems(int schemeId, LocalDateTime newStart, LocalDateTime newEnd, int updatedBy)
1647
            throws ProfitMandiBusinessException {
1648
        List<SchemeItem> outsideItems = schemeItemRepository.selectItemsOutsideWindow(schemeId, newStart, newEnd);
1649
        List<SchemeItem> clamped = new ArrayList<>();
1650
 
1651
        for (SchemeItem item : outsideItems) {
1652
            LocalDateTime clampedStart = item.getStartDate().isBefore(newStart) ? newStart : item.getStartDate();
1653
            LocalDateTime clampedEnd = item.getEndDate().isAfter(newEnd) ? newEnd : item.getEndDate();
1654
 
1655
            if (clampedStart.isAfter(clampedEnd) || clampedStart.isEqual(clampedEnd)) {
1656
                throw new ProfitMandiBusinessException("schemeId, catalogId",
1657
                        schemeId + ", " + item.getCatalogId(),
1658
                        "Cannot clamp: item window for catalogId " + item.getCatalogId()
1659
                                + " (" + FormattingUtils.formatDate(item.getStartDate()) + " - " + FormattingUtils.formatDate(item.getEndDate()) + ")"
1660
                                + " falls entirely outside new scheme window ("
1661
                                + FormattingUtils.formatDate(newStart) + " - " + FormattingUtils.formatDate(newEnd)
1662
                                + "). Adjust this item first.");
1663
            }
1664
 
1665
            item.setStartDate(clampedStart);
1666
            item.setEndDate(clampedEnd);
1667
            item.setUpdatedBy(updatedBy);
1668
            item.setUpdatedOn(LocalDateTime.now());
1669
            schemeItemRepository.persist(item);
1670
            clamped.add(item);
1671
        }
1672
        return clamped;
1673
    }
22653 ashik.ali 1674
}