Subversion Repositories SmartDukaan

Rev

Rev 34698 | Rev 34706 | Go to most recent revision | 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;
34705 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.*;
34705 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));
34705 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
 
34705 amit.gupta 77
 
34317 amit.gupta 78
    private static final List<SchemeType> ACTIVATION_SCHEME_TYPES = Arrays.asList(SchemeType.SPECIAL_SUPPORT, SchemeType.SELLOUT);
33600 ranu 79
 
31410 amit.gupta 80
    @Autowired
34221 tejus.loha 81
    SchemeBlockedImeiRepository schemeBlockedImeiRepository;
82
    @Autowired
31410 amit.gupta 83
    StateGstRateRepository stateGstRateRepository;
84
    @Autowired
85
    NotificationService notificationService;
86
    @Autowired
87
    @Qualifier("fofoInventoryItemRepository")
88
    private InventoryItemRepository inventoryItemRepository;
89
    @Autowired
90
    private ActivatedImeiRepository activatedImeiRepository;
91
    @Autowired
92
    private PartnerTypeChangeService partnerTypeChangeService;
93
    @Autowired
94
    private PurchaseService purchaseService;
95
    @Autowired
96
    private ScanRecordRepository scanRecordRepository;
97
    @Autowired
98
    private SessionFactory sessionFactory;
99
    @Autowired
100
    private SchemeRepository schemeRepository;
101
    @Autowired
102
    private PriceDropRepository priceDropRepository;
103
    @Autowired
104
    private RoleManager roleManager;
105
    @Autowired
106
    private RetailerRepository retailerRepository;
107
    @Autowired
108
    private ReporticoService reporticoService;
109
    @Autowired
110
    private TagListingRepository tagListingRepository;
111
    @Autowired
112
    private SchemeInOutRepository schemeInOutRepository;
113
    @Autowired
114
    @Qualifier("catalogItemRepository")
115
    private ItemRepository itemRepository;
116
    @Autowired
117
    private SchemeItemRepository schemeItemRepository;
118
    @Autowired
119
    private SchemeRegionRepository schemeRegionRepository;
120
    @Autowired
121
    private WalletService walletService;
122
    @Autowired
123
    private PurchaseRepository purchaseRepository;
124
    @Autowired
125
    private FofoOrderRepository fofoOrderRepository;
31921 amit.gupta 126
    @Autowired
127
    private PriceCircularService priceCircularService;
33600 ranu 128
    @Autowired
129
    private SamsungExceptionRepository samsungExceptionRepository;
130
 
31410 amit.gupta 131
    @Override
34221 tejus.loha 132
    public List<String> getBlockedImeis() {
34254 tejus.loha 133
        List<SchemeBlockedImei> schemeBlockedImeis = schemeBlockedImeiRepository.selectAll();
34221 tejus.loha 134
        return schemeBlockedImeis.stream().filter(x -> x.isBlocked()).map(x -> x.getImei()).collect(Collectors.toList());
135
    }
136
 
137
    @Override
31410 amit.gupta 138
    public void saveScheme(int creatorId, CreateSchemeRequest createSchemeRequest) throws ProfitMandiBusinessException {
23444 amit.gupta 139
 
31410 amit.gupta 140
        this.validateCreateSchemeRequest(createSchemeRequest);
27898 amit.gupta 141
 
31410 amit.gupta 142
        Scheme scheme = this.toScheme(creatorId, createSchemeRequest);
25503 amit.gupta 143
 
31410 amit.gupta 144
        if (scheme.getStartDateTime().isAfter(scheme.getEndDateTime())) {
145
            throw new ProfitMandiBusinessException(
146
                    ProfitMandiConstants.START_DATE + ", " + ProfitMandiConstants.END_DATE,
147
                    scheme.getStartDateTime() + ", " + scheme.getEndDateTime(), "SCHM_VE_1005");
148
        }
26722 amit.gupta 149
 
31410 amit.gupta 150
        // this.validateItemIds(createSchemeRequest);
151
        schemeRepository.persist(scheme);
152
        for (int catalogId : createSchemeRequest.getCatalogIds()) {
153
            SchemeItem schemeItem = new SchemeItem();
154
            schemeItem.setSchemeId(scheme.getId());
155
            schemeItem.setCatalogId(catalogId);
34317 amit.gupta 156
            schemeItem.setStartDate(createSchemeRequest.getStartDate());
157
            schemeItem.setEndDate(createSchemeRequest.getEndDate());
31410 amit.gupta 158
            schemeItemRepository.persist(schemeItem);
159
        }
23444 amit.gupta 160
 
31410 amit.gupta 161
        for (int regionId : createSchemeRequest.getRegionIds()) {
162
            SchemeRegion schemeRegion = new SchemeRegion();
163
            schemeRegion.setSchemeId(scheme.getId());
164
            schemeRegion.setRegionId(regionId);
165
            schemeRegionRepository.persist(schemeRegion);
166
        }
26332 amit.gupta 167
 
31410 amit.gupta 168
    }
26722 amit.gupta 169
 
31410 amit.gupta 170
    private void validateCreateSchemeRequest(CreateSchemeRequest createSchemeRequest)
171
            throws ProfitMandiBusinessException {
172
        if (createSchemeRequest.getName() == null || createSchemeRequest.getName().isEmpty()) {
173
            throw new ProfitMandiBusinessException(ProfitMandiConstants.NAME, createSchemeRequest.getName(),
174
                    "SCHM_VE_1000");
175
        }
176
        if (createSchemeRequest.getAmount() <= 0) {
177
            throw new ProfitMandiBusinessException(ProfitMandiConstants.AMOUNT, createSchemeRequest.getAmount(),
178
                    "SCHM_VE_1001");
179
        }
26684 amit.gupta 180
 
31410 amit.gupta 181
        if (createSchemeRequest.getAmountType().equals(AmountType.PERCENTAGE)
182
                && createSchemeRequest.getAmount() > 100) {
183
            throw new ProfitMandiBusinessException(ProfitMandiConstants.AMOUNT, createSchemeRequest.getAmount(),
184
                    "SCHM_VE_1002");
185
        }
23444 amit.gupta 186
 
31410 amit.gupta 187
        if (createSchemeRequest.getStartDate() == null) {
188
            throw new ProfitMandiBusinessException(ProfitMandiConstants.START_DATE, createSchemeRequest.getStartDate(),
189
                    "SCHM_VE_1003");
190
        }
191
        if (createSchemeRequest.getEndDate() == null) {
192
            throw new ProfitMandiBusinessException(ProfitMandiConstants.END_DATE, createSchemeRequest.getEndDate(),
193
                    "SCHM_VE_1004");
194
        }
195
    }
23798 amit.gupta 196
 
31410 amit.gupta 197
    private void validateItemIds(CreateSchemeRequest createSchemeRequest) throws ProfitMandiBusinessException {
198
        if (createSchemeRequest.getCatalogIds() == null || createSchemeRequest.getCatalogIds().isEmpty()) {
199
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ITEM_ID, createSchemeRequest.getCatalogIds(),
200
                    "SCHM_1003");
201
        }
202
        List<Integer> foundItemIds = itemRepository.selectIdsByIdsAndType(createSchemeRequest.getCatalogIds(),
203
                ItemType.SERIALIZED);
204
        if (foundItemIds.size() != createSchemeRequest.getCatalogIds().size()) {
205
            createSchemeRequest.getCatalogIds().removeAll(foundItemIds);
206
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ITEM_ID, createSchemeRequest.getCatalogIds(),
207
                    "SCHM_1004");
208
        }
209
    }
29927 amit.gupta 210
 
31410 amit.gupta 211
    @Override
212
    public Scheme getSchemeById(int schemeId) throws ProfitMandiBusinessException {
213
        Scheme scheme = schemeRepository.selectById(schemeId);
214
        List<Integer> catalogIds = schemeItemRepository.selectCatalogIdsBySchemeId(scheme.getId());
215
        if (catalogIds.size() > 0) {
216
            List<Item> items = itemRepository.selectAllByCatalogIds(new HashSet<>(catalogIds));
217
            scheme.setCatalogStringMap(this.toCatalogStringMap(items));
218
        }
219
        return scheme;
220
    }
24562 amit.gupta 221
 
31410 amit.gupta 222
    public Map<Integer, String> toCatalogStringMap(List<Item> items) {
223
        Map<Integer, String> catalogMap = new HashMap<>();
224
        for (Item item : items) {
225
            if (!catalogMap.containsKey(item.getCatalogItemId())) {
226
                catalogMap.put(item.getCatalogItemId(), item.getItemDescriptionNoColor());
227
            }
228
        }
229
        return catalogMap;
230
    }
29927 amit.gupta 231
 
31410 amit.gupta 232
    private Set<Integer> schemeItemsToCatalogIds(List<SchemeItem> schemeItems) {
233
        Set<Integer> catalogId = new HashSet<>();
234
        for (SchemeItem schemeItem : schemeItems) {
235
            catalogId.add(schemeItem.getCatalogId());
236
        }
237
        return catalogId;
238
    }
23444 amit.gupta 239
 
31410 amit.gupta 240
    @Override
33248 ranu 241
    public List<SchemeModel> getAllSchemeModels(LocalDateTime startDateTime, LocalDateTime endDateTime) throws ProfitMandiBusinessException {
31410 amit.gupta 242
        List<Scheme> schemes = schemeRepository.selectAllBetweenCreateTimestamp(startDateTime, endDateTime);
243
        Map<Integer, Scheme> schemeIdSchemeMap = schemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
244
        List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIds(schemeIdSchemeMap.keySet());
245
        Set<Integer> catalogIds = schemeItems.stream().map(x -> x.getCatalogId()).collect(Collectors.toSet());
246
        List<Item> items = itemRepository.selectAllByCatalogIds(catalogIds);
247
        Map<Integer, String> catalogStringMap = this.toCatalogStringMap(items);
248
        this.addCatalogIdsToSchemes(schemeItems, schemeIdSchemeMap, catalogStringMap);
249
        return this.toSchemeModels(schemeIdSchemeMap);
250
    }
23444 amit.gupta 251
 
31410 amit.gupta 252
    private List<SchemeModel> toSchemeModels(Map<Integer, Scheme> schemeIdSchemeMap) {
253
        List<SchemeModel> schemeModels = new ArrayList<>();
254
        for (Map.Entry<Integer, Scheme> schemeIdSchemeEntry : schemeIdSchemeMap.entrySet()) {
255
            schemeModels.add(this.toSchemeModel(schemeIdSchemeEntry.getValue()));
256
        }
257
        return schemeModels;
258
    }
23444 amit.gupta 259
 
31410 amit.gupta 260
    private SchemeModel toSchemeModel(Scheme scheme) {
261
        SchemeModel schemeModel = new SchemeModel();
262
        schemeModel.setSchemeId(scheme.getId());
263
        schemeModel.setName(scheme.getName());
264
        schemeModel.setDescription(scheme.getDescription());
265
        schemeModel.setSchemeType(scheme.getType().toString());
266
        schemeModel.setAmountType(scheme.getAmountType().toString());
267
        schemeModel.setAmount(scheme.getAmount());
268
        schemeModel.setStartDateTime(StringUtils.toString(scheme.getStartDateTime()));
269
        schemeModel.setEndDateTime(StringUtils.toString(scheme.getEndDateTime()));
270
        schemeModel.setCreateTimestamp(StringUtils.toString(scheme.getCreateTimestamp()));
271
        schemeModel.setActiveTimestamp(StringUtils.toString(scheme.getActiveTimestamp()));
272
        schemeModel.setExpireTimestamp(StringUtils.toString(scheme.getExpireTimestamp()));
273
        schemeModel.setCreatedBy(scheme.getCreatedBy());
274
        schemeModel.setCatalogStringMap(scheme.getCatalogStringMap());
275
        schemeModel.setRetailerIds(scheme.getRetailerIds());
276
        return schemeModel;
277
    }
23444 amit.gupta 278
 
31410 amit.gupta 279
    private void addCatalogIdsToSchemes(List<SchemeItem> schemeItems, Map<Integer, Scheme> schemeIdSchemeMap,
280
                                        Map<Integer, String> catalogStringMap) {
281
        for (SchemeItem schemeItem : schemeItems) {
282
            Scheme scheme = schemeIdSchemeMap.get(schemeItem.getSchemeId());
283
            scheme.getCatalogStringMap().put(schemeItem.getCatalogId(), catalogStringMap.get(schemeItem.getCatalogId()));
284
        }
285
    }
31170 amit.gupta 286
 
31410 amit.gupta 287
    @Override
288
    public void activeSchemeById(int schemeId) throws ProfitMandiBusinessException {
289
        Scheme scheme = schemeRepository.selectById(schemeId);
290
        if (scheme.getActiveTimestamp() != null) {
291
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ACTIVE_TIMESTAMP, scheme.getActiveTimestamp(),
292
                    "SCHM_1005");
293
        }
294
        if (scheme.getExpireTimestamp() != null) {
295
            throw new ProfitMandiBusinessException(ProfitMandiConstants.EXPIRE_TIMESTAMP, scheme.getExpireTimestamp(),
296
                    "SCHM_1006");
297
        }
298
        scheme.setActiveTimestamp(LocalDateTime.now());
299
        this.sendSchemeNotification(scheme);
23444 amit.gupta 300
 
301
 
31410 amit.gupta 302
        /*
303
         * if (scheme.getType() == SchemeType.IN) {
304
         * this.processPreviousPurchases(scheme); } else if (scheme.getType() ==
305
         * SchemeType.OUT) { this.processPreviousSales(scheme); }
306
         */
307
    }
23796 amit.gupta 308
 
34568 vikas.jang 309
    @Override
310
    public void activeSchemeByIds(List<Scheme> schemes) throws ProfitMandiBusinessException {
311
        if (schemes.size() > 0) {
312
            for (Scheme scheme : schemes) {
313
                if (scheme.getActiveTimestamp() != null || scheme.getExpireTimestamp() != null) {
314
                    continue;
315
                }
316
                scheme.setActiveTimestamp(LocalDateTime.now());
317
            }
318
            this.sendCombinedSchemesNotification(schemes);
319
        }
320
    }
321
 
31410 amit.gupta 322
    private void sendSchemeNotification(Scheme scheme) throws ProfitMandiBusinessException {
323
        if (ACTIVATION_SCHEME_TYPES.contains(scheme.getType())) {
31336 amit.gupta 324
 
31410 amit.gupta 325
            SendNotificationModel sendNotificationModel = new SendNotificationModel();
326
            List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIds(Collections.singleton(scheme.getId()));
327
            Set<Integer> catalogIds = schemeItems.stream().map(x -> x.getCatalogId()).collect(Collectors.toSet());
328
            List<String> itemDescriptions = itemRepository.selectAllByCatalogIds(catalogIds).stream().filter(Utils.distinctByKey(Item::getCatalogItemId))
329
                    .map(x -> x.getItemDescriptionNoColor()).collect(Collectors.toList());
330
            String titleTemplate = "%s of Rs.%s for %s";
331
            String schemeString = "Activation scheme";
332
            if (scheme.getType().equals(SchemeType.SPECIAL_SUPPORT)) {
333
                schemeString = "Special Support";
334
            }
30957 amit.gupta 335
 
31410 amit.gupta 336
            String message = "Duration from - " + FormattingUtils.formatDateMonth(scheme.getStartDateTime()) + " - " + FormattingUtils.formatDateMonth(scheme.getEndDateTime());
337
            sendNotificationModel.setCampaignName("activationscheme");
32602 amit.gupta 338
            sendNotificationModel.setUrl("https://app.smartdukaan.com/pages/home/notifications");
31410 amit.gupta 339
            sendNotificationModel.setExpiresat(LocalDateTime.now().plusDays(1));
340
            sendNotificationModel.setMessage(message);
341
            sendNotificationModel.setTitle(String.format(titleTemplate, schemeString, FormattingUtils.formatDecimal(scheme.getAmount()), org.apache.commons.lang3.StringUtils.abbreviate(String.join(", ", itemDescriptions), 25)));
342
            sendNotificationModel.setType("url");
343
            sendNotificationModel.setMessageType(MessageType.scheme);
344
            notificationService.sendNotificationToAll(sendNotificationModel);
345
        }
346
    }
34705 amit.gupta 347
    @Autowired
348
    LoanRepository loanRepository;
23444 amit.gupta 349
 
31410 amit.gupta 350
    @Override
351
    public void expireSchemeById(int schemeId, LocalDateTime expiryTime) throws ProfitMandiBusinessException {
352
        Scheme scheme = schemeRepository.selectById(schemeId);
353
        if (scheme == null || scheme.getActiveTimestamp() == null) {
354
            throw new ProfitMandiBusinessException(ProfitMandiConstants.ACTIVE_TIMESTAMP, scheme.getActiveTimestamp(),
355
                    "SCHM_1007");
356
        }
357
        if (scheme.getExpireTimestamp() != null) {
358
            throw new ProfitMandiBusinessException(ProfitMandiConstants.EXPIRE_TIMESTAMP, scheme.getExpireTimestamp(),
359
                    "SCHM_1008");
360
        }
361
        scheme.setExpireTimestamp(LocalDateTime.now());
362
        if (expiryTime.isAfter(scheme.getEndDateTime())) {
363
            throw new ProfitMandiBusinessException(ProfitMandiConstants.EXPIRE_TIMESTAMP, scheme.getExpireTimestamp(),
364
                    "End Date cant be extended during expiry");
365
        }
366
        scheme.setEndDateTime(expiryTime);
367
        schemeRepository.persist(scheme);
368
    }
23444 amit.gupta 369
 
31410 amit.gupta 370
    private Map<Integer, Scheme> toSchemeIdSchemeMap(List<Scheme> schemes) {
371
        Map<Integer, Scheme> schemeIdSchemeMap = new HashMap<>();
372
        for (Scheme scheme : schemes) {
373
            schemeIdSchemeMap.put(scheme.getId(), scheme);
374
        }
375
        return schemeIdSchemeMap;
376
    }
23444 amit.gupta 377
 
31410 amit.gupta 378
    private Map<Integer, Set<Scheme>> toCatalogIdSchemesMap(List<SchemeItem> schemeItems, List<Scheme> schemes) {
379
        Map<Integer, Scheme> schemesMap = schemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
380
        Map<Integer, Set<Scheme>> catalogSchemesMap = new HashMap<>();
381
        for (SchemeItem schemeItem : schemeItems) {
382
            if (!catalogSchemesMap.containsKey(schemeItem.getCatalogId())) {
383
                catalogSchemesMap.put(schemeItem.getCatalogId(), new HashSet<>());
384
            }
385
            Set<Scheme> schemesSet = catalogSchemesMap.get(schemeItem.getCatalogId());
386
            schemesSet.add(schemesMap.get(schemeItem.getSchemeId()));
387
        }
388
        return catalogSchemesMap;
389
    }
23444 amit.gupta 390
 
31410 amit.gupta 391
    private Map<InventoryItem, Set<Scheme>> toInventoryItemSchemesMap(List<Scheme> schemes,
33248 ranu 392
                                                                      List<InventoryItem> inventoryItems) throws ProfitMandiBusinessException {
31410 amit.gupta 393
        Set<Integer> schemeIds = schemes.stream().map(x -> x.getId()).collect(Collectors.toSet());
394
        Set<Integer> itemIds = inventoryItems.stream().map(x -> x.getItemId()).collect(Collectors.toSet());
395
        Set<Integer> catalogIds = itemRepository.selectByIds(itemIds).stream().map(x -> x.getCatalogItemId()).collect(Collectors.toSet());
396
        List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIdsAndCatalogIds(schemeIds, catalogIds);
23444 amit.gupta 397
 
31410 amit.gupta 398
        Map<Integer, Set<Scheme>> catalogIdSchemesMap = this.toCatalogIdSchemesMap(schemeItems, schemes);
399
        Map<InventoryItem, Set<Scheme>> inventoryItemSchemesMap = new HashMap<>();
400
        for (InventoryItem inventoryItem : inventoryItems) {
401
            LOGGER.info("inventoryItem {}", inventoryItem);
402
            LOGGER.info("inventoryItem.getItem() {}", inventoryItem.getItem());
403
            LOGGER.info("catalogIdSchemesMap {}", catalogIdSchemesMap);
404
            if (catalogIdSchemesMap.containsKey(inventoryItem.getItem().getCatalogItemId())) {
405
                inventoryItemSchemesMap.put(inventoryItem, catalogIdSchemesMap.get(inventoryItem.getItem().getCatalogItemId()));
406
            }
407
        }
408
        return inventoryItemSchemesMap;
409
    }
31170 amit.gupta 410
 
31903 amit.gupta 411
    @Autowired
412
    OfferTargetSlabRepository offerTargetSlabRepository;
413
 
31410 amit.gupta 414
    private Scheme toScheme(int creatorId, CreateSchemeRequest createSchemeRequest) {
415
        Scheme scheme = new Scheme();
416
        scheme.setName(createSchemeRequest.getName());
417
        scheme.setDescription(createSchemeRequest.getDescription());
418
        scheme.setType(createSchemeRequest.getType());
419
        scheme.setAmountType(createSchemeRequest.getAmountType());
420
        scheme.setAmount(createSchemeRequest.getAmount());
421
        scheme.setPartnerType(createSchemeRequest.getPartnerType());
422
        scheme.setStartDateTime(createSchemeRequest.getStartDate());
423
        scheme.setEndDateTime(createSchemeRequest.getEndDate());
424
        scheme.setCreatedBy(creatorId);
425
        scheme.setCashback(createSchemeRequest.isCashback());
426
        return scheme;
427
    }
23444 amit.gupta 428
 
31903 amit.gupta 429
    @Autowired
430
    OfferRepository offerRepository;
431
    @Autowired
432
    OfferPayoutRepository offerPayoutRepository;
433
    @Autowired
434
    AgeingService ageingService;
31588 amit.gupta 435
 
34705 amit.gupta 436
    private void sendCombinedSchemesNotification(List<Scheme> schemes) throws ProfitMandiBusinessException {
437
        if (schemes == null || schemes.isEmpty()) return;
438
 
439
        SendNotificationModel sendNotificationModel = new SendNotificationModel();
440
        StringBuilder messageBuilder = new StringBuilder();
441
        StringBuilder titleBuilder = new StringBuilder();
442
        Set<Integer> allCatalogIds = new HashSet<>();
443
 
444
        for (Scheme scheme : schemes) {
445
            if (!ACTIVATION_SCHEME_TYPES.contains(scheme.getType())) {
446
                continue;
447
            }
448
 
449
            String schemeTypeLabel = "Activation scheme";
450
            if (scheme.getType().equals(SchemeType.SPECIAL_SUPPORT)) {
451
                schemeTypeLabel = "Special Support";
452
            }
453
 
454
            List<SchemeItem> schemeItems = schemeItemRepository.selectBySchemeIds(Collections.singleton(scheme.getId()));
455
            Set<Integer> catalogIds = schemeItems.stream().map(SchemeItem::getCatalogId).collect(Collectors.toSet());
456
            allCatalogIds.addAll(catalogIds);
457
 
458
            List<String> itemBrand = itemRepository.selectAllByCatalogIds(catalogIds).stream()
459
                    .filter(Utils.distinctByKey(Item::getCatalogItemId))
460
                    .map(Item::getBrand)
461
                    .collect(Collectors.toList());
462
 
463
            List<String> itemDescriptions = itemRepository.selectAllByCatalogIds(catalogIds).stream()
464
                    .filter(Utils.distinctByKey(Item::getCatalogItemId))
465
                    .map(Item::getItemDescriptionNoColor)
466
                    .collect(Collectors.toList());
467
 
468
            // Title line
469
            titleBuilder.append(String.format("%s of Rs.%s for %s\n",
470
                    schemeTypeLabel,
471
                    FormattingUtils.formatDecimal(scheme.getAmount()),
472
                    itemBrand.get(0)));
473
 
474
            // Message details
475
            messageBuilder.append(String.format("On Selected Models: %s\n", String.join(", ", itemDescriptions)));
476
            messageBuilder.append(String.format("Duration from - %s to %s\n\n",
477
                    FormattingUtils.formatDateMonth(scheme.getStartDateTime()),
478
                    FormattingUtils.formatDateMonth(scheme.getEndDateTime())));
479
        }
480
 
481
        if (titleBuilder.length() == 0) return;
482
 
483
        sendNotificationModel.setCampaignName("activationscheme");
484
        sendNotificationModel.setUrl("https://app.smartdukaan.com/pages/home/notifications");
485
        sendNotificationModel.setExpiresat(LocalDateTime.now().plusDays(1));
486
        sendNotificationModel.setTitle(titleBuilder.toString().trim());
487
        sendNotificationModel.setMessage(messageBuilder.toString().trim());
488
        sendNotificationModel.setType("url");
489
        sendNotificationModel.setMessageType(MessageType.scheme);
490
        notificationService.sendNotificationToAll(sendNotificationModel);
491
    }
492
 
493
    //CDs would be rejected based on cdFreeDays
31903 amit.gupta 494
    public void processSchemeIn(int purchaseId, int retailerId) throws ProfitMandiBusinessException {
34504 amit.gupta 495
        LOGGER.info("Processing scheme in  for purchaseId - {}", purchaseId);
31903 amit.gupta 496
        Purchase purchase = purchaseRepository.selectByIdAndFofoId(purchaseId, retailerId);
34705 amit.gupta 497
        String purchaseInvoice = purchase.getPurchaseReference();
498
 
499
        Loan loan = loanRepository.selectLoanByInvoice(purchaseInvoice);
500
 
34317 amit.gupta 501
        PartnerType partnerType = partnerTypeChangeService.getTypeOnMonth(retailerId,
502
                YearMonth.from(purchase.getCreateTimestamp()));
34139 amit.gupta 503
        List<InventoryItem> inventoryItems = inventoryItemRepository.selectByPurchaseId(purchaseId);
31903 amit.gupta 504
        //Remove imeis from blocked imeis list
34317 amit.gupta 505
        List<String> blockedImeis = this.getBlockedImeis();
506
        inventoryItems = inventoryItems.stream().filter(inventoryItem -> !blockedImeis.contains(inventoryItem.getSerialNumber())).collect(Collectors.toList());
31903 amit.gupta 507
        if (inventoryItems.size() == 0) return;
23444 amit.gupta 508
 
31903 amit.gupta 509
        Set<Integer> itemIds = inventoryItems.stream().map(x -> x.getItemId()).collect(Collectors.toSet());
510
        Map<Integer, Item> itemsMap = itemRepository.selectByIds(itemIds).stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
511
        inventoryItems.stream().forEach(x -> x.setItem(itemsMap.get(x.getItemId())));
23444 amit.gupta 512
 
513
 
34317 amit.gupta 514
        LocalDateTime billingDate = purchaseService.getBillingDateOfPurchase(purchaseId);
31903 amit.gupta 515
        Set<Integer> itemIdsSet = tagListingRepository.selectByItemIdsAndTagIds(itemIds, tagIds).stream()
516
                .filter(x -> x.getEolDate() == null || x.getEolDate().isAfter(billingDate)).map(x -> x.getItemId())
517
                .collect(Collectors.toSet());
518
        // Only consider inventory items that were not returned and not eol
519
        inventoryItems = inventoryItems.stream().filter(x -> itemIdsSet.contains(x.getItemId()))
520
                .filter(x -> !x.getLastScanType().equals(ScanType.PURCHASE_RET_BAD))
521
                .filter(x -> !x.getLastScanType().equals(ScanType.PURCHASE_RET)).collect(Collectors.toList());
522
 
523
        if (inventoryItems.size() == 0) return;
524
 
525
        Map<Integer, List<InventoryItem>> catalogInventoryItemMap = inventoryItems.stream().collect(Collectors.groupingBy(x -> x.getItem().getCatalogItemId()));
34317 amit.gupta 526
        Map<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSchemeSummaryMap = tagListingRepository.getModelSchemesByCatalogIdsAndType(retailerId,
32454 amit.gupta 527
                partnerType, new ArrayList<>(catalogInventoryItemMap.keySet()), billingDate);
31903 amit.gupta 528
 
529
 
530
        int itemsCount = 0;
531
        float totalCashback = 0;
34698 amit.gupta 532
        //LOGGER.info("catalogSchemeSummaryMap - {}", catalogSchemeSummaryMap);
31903 amit.gupta 533
        for (Map.Entry<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSummaryModelListEntry : catalogSchemeSummaryMap.entrySet()) {
534
            CatalogSummaryModel catalogSummaryModel = catalogSummaryModelListEntry.getKey();
34504 amit.gupta 535
            List<SchemeSummaryModel> schemeSummaryModels = catalogSummaryModelListEntry.getValue().stream().filter(Objects::nonNull).collect(Collectors.toList());
34317 amit.gupta 536
            schemeSummaryModels.stream().filter(x -> x != null && x.getSchemeType().getTransactionType().equals(StockTransactionType.IN)).forEach(x -> x.setProcess(true));
31903 amit.gupta 537
            if (schemeSummaryModels.stream().filter(x -> x.isProcess()).count() == 0) continue;
538
            List<InventoryItem> modelInventoryItems = catalogInventoryItemMap.get(catalogSummaryModel.getCatalogId());
539
            for (InventoryItem inventoryItem : modelInventoryItems) {
32165 amit.gupta 540
                float inventoryItemCashback = this.createSchemeInOut(schemeSummaryModels, inventoryItem);
34705 amit.gupta 541
                if (loan != null && loan.getCdFreeDays() > 0) {
542
                    List<SchemeType> rejectedCDTypes = HALF_DAYS_CD_REJECT_SCHEME_TYPES;
543
                    if (loan.getCdFreeDays() == ProfitMandiConstants.LOAN_FULL_CREDIT_DAYS) {
544
                        rejectedCDTypes = FULL_DAYS_CD_SCHEME_TYPES;
545
                    }
546
                    for (SchemeType rejectedCDType : rejectedCDTypes) {
547
                        SchemeInOut sio = schemeInOutRepository.selectAllByType(rejectedCDType, inventoryItem.getId()).get(0);
548
                        sio.setStatus(SchemePayoutStatus.REJECTED);
549
                        sio.setStatusDescription("Rejected due to free days availed");
550
                        sio.setRolledBackTimestamp(LocalDateTime.now());
551
                        inventoryItemCashback -= sio.getAmount();
552
                    }
553
 
554
                }
555
                if (inventoryItemCashback > 0.01f) {
31903 amit.gupta 556
                    itemsCount++;
557
                    totalCashback += inventoryItemCashback;
31410 amit.gupta 558
                }
559
            }
31903 amit.gupta 560
        }
22653 ashik.ali 561
 
29927 amit.gupta 562
 
34317 amit.gupta 563
        LOGGER.info("Items count for purchase id {} is {}", purchaseId, itemsCount);
31903 amit.gupta 564
        if (itemsCount > 0) {
565
            walletService.addAmountToWallet(
34317 amit.gupta 566
                    retailerId, purchaseId, WalletReferenceType.SCHEME_IN, "Added for SCHEME IN against invoice "
31903 amit.gupta 567
                            + purchase.getPurchaseReference() + " (total " + itemsCount + " pcs)",
568
                    totalCashback, purchase.getCreateTimestamp());
569
            LOGGER.info("Added Rs.{} for SCHEME IN against invoice {} total pcs({}) {}", totalCashback,
570
                    purchase.getPurchaseReference(), itemsCount);
34504 amit.gupta 571
        }
34317 amit.gupta 572
 
31903 amit.gupta 573
    }
574
 
575
 
31410 amit.gupta 576
    // We are maintaining price drop after grn
577
    private float getAmount(InventoryItem inventoryItem, Scheme scheme) throws ProfitMandiBusinessException {
34221 tejus.loha 578
        if (this.getBlockedImeis().contains(inventoryItem.getSerialNumber())) {
31410 amit.gupta 579
            return 0;
580
        }
581
        float amount = 0;
582
        float dpForCalc = 0;
583
        float taxableSellingPrice = 0;
29927 amit.gupta 584
 
31410 amit.gupta 585
        //float totalTaxRate = stateGstRateRepository.getTotalTaxRate(inventoryItem.getItemId());
586
        if (scheme.getAmountType().equals(AmountType.PERCENTAGE)) {
34317 amit.gupta 587
            if (scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
31410 amit.gupta 588
                dpForCalc = inventoryItem.getUnitPrice() - inventoryItem.getPriceDropAmount();
589
            } else {
590
                try {
591
                    dpForCalc = Math.min(inventoryItem.getUnitPrice() - inventoryItem.getPriceDropAmount(),
592
                            tagListingRepository.selectByItemId(inventoryItem.getItemId()).getSellingPrice());
593
                } catch (Exception e) {
594
                    LOGGER.info("Could not find tag Listing entry in {}", inventoryItem.getItemId());
595
                    e.printStackTrace();
596
                }
597
            }
598
            //TODO:Should be calculated on unit price
599
            //taxableSellingPrice = dpForCalc / (1 + totalTaxRate / 100);
600
            //amount = taxableSellingPrice * scheme.getAmount() / 100;
601
            amount = dpForCalc * scheme.getAmount() / 100;
602
            System.out.println(String.format("%d\t%s\t%d\t%d\t%s\t%s\t%s\t%s\t%f\t%f\t%f\t%f", inventoryItem.getId(),
603
                    inventoryItem.getSerialNumber(), inventoryItem.getItemId(), scheme.getId(), scheme.getName(),
604
                    scheme.getType(), scheme.getAmountType(), scheme.getPartnerType(), dpForCalc, taxableSellingPrice,
605
                    scheme.getAmount(), amount));
34317 amit.gupta 606
        } else if (scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
31410 amit.gupta 607
            amount = scheme.getAmount();
608
        }
609
        return amount;
610
    }
23444 amit.gupta 611
 
31903 amit.gupta 612
    //Only in and activation margins are allowed to be rolled out more than twice
32165 amit.gupta 613
    private float createSchemeInOut(List<SchemeSummaryModel> schemeSummaryModels, InventoryItem inventoryItem) throws ProfitMandiBusinessException {
33614 amit.gupta 614
        LOGGER.info("schemeSummaryModels - {}", schemeSummaryModels);
31921 amit.gupta 615
        InventoryPayoutModel inventoryPayoutModel = priceCircularService.getPayouts(inventoryItem);
31903 amit.gupta 616
        //Get all schemes
617
        List<SchemeSummaryModel> inventoryPayoutModelToProcess = schemeSummaryModels.stream().filter(x -> x.isProcess()).collect(Collectors.toList());
34698 amit.gupta 618
        LOGGER.info("inventoryPayoutModel - {}", inventoryPayoutModel);
32165 amit.gupta 619
        List<SchemeInOut> paidSios = inventoryPayoutModel.getPaidSios();
620
        List<SchemeInOut> pendingSios = inventoryPayoutModel.getPendingSios();
621
        Map<SchemeType, List<SchemeInOut>> paidSchemeTypesMap = inventoryPayoutModel.getPaidSios().stream().collect(Collectors.groupingBy(x -> x.getScheme().getType()));
622
        Map<Integer, SchemeInOut> paidSchemesMap = paidSios.stream().collect(Collectors.toMap(x -> x.getSchemeId(), x -> x));
623
        Map<Integer, SchemeInOut> pendingSchemesMap = pendingSios.stream().collect(Collectors.toMap(x -> x.getSchemeId(), x -> x));
32155 amit.gupta 624
        Map<SchemeType, Float> schemeTypeCancelledAmountMap = new HashMap<>();
31903 amit.gupta 625
 
626
        double percentageToPay = 0d;
627
        double fixedToPay = 0d;
628
        Map<SchemeSummaryModel, AmountModel> payoutSchemeSummaryModelMap = new HashMap<>();
629
        for (SchemeSummaryModel schemeSummaryModelToProcess : inventoryPayoutModelToProcess) {
32972 amit.gupta 630
            if (paidSchemesMap.containsKey(schemeSummaryModelToProcess.getSchemeId()) || pendingSchemesMap.containsKey(schemeSummaryModelToProcess.getSchemeId()))
631
                continue;
31903 amit.gupta 632
            //If different type but on higher side provide the higher margin
34317 amit.gupta 633
            if (Arrays.asList(SchemeType.SPECIAL_SUPPORT).contains(schemeSummaryModelToProcess.getSchemeType())) {
32165 amit.gupta 634
                Scheme scheme = schemeRepository.selectById(schemeSummaryModelToProcess.getSchemeId());
635
                //Create only if the activation if the activation date is not known
636
                ActivatedImei activatedImei = activatedImeiRepository.selectBySerialNumber(inventoryItem.getSerialNumber());
32972 amit.gupta 637
                if (activatedImei == null || activatedImei.getActivationTimestamp() == null) {
32165 amit.gupta 638
                    SchemeInOut sio = new SchemeInOut();
639
                    sio.setAmount(0);
640
                    sio.setInventoryItemId(inventoryItem.getId());
641
                    sio.setSchemeId(schemeSummaryModelToProcess.getSchemeId());
642
                    sio.setStatusDescription("Activation pending for IMEI#" + inventoryItem.getSerialNumber());
643
                    sio.setStatus(SchemePayoutStatus.PENDING);
644
                    schemeInOutRepository.persist(sio);
645
                }
34317 amit.gupta 646
            } else if (!StockTransactionType.IN.equals(schemeSummaryModelToProcess.getSchemeType().getTransactionType())) {
31903 amit.gupta 647
                //We have got non repeating scheme type
34705 amit.gupta 648
 
649
                if (schemeSummaryModelToProcess.getAmountType().equals(AmountType.PERCENTAGE)) {
650
                    percentageToPay += schemeSummaryModelToProcess.getAmount();
31903 amit.gupta 651
                } else {
34705 amit.gupta 652
                    fixedToPay += schemeSummaryModelToProcess.getAmount();
31903 amit.gupta 653
                }
34705 amit.gupta 654
                payoutSchemeSummaryModelMap.put(schemeSummaryModelToProcess, new AmountModel(schemeSummaryModelToProcess.getAmount(), schemeSummaryModelToProcess.getAmountType()));
655
 
31903 amit.gupta 656
            } else {
657
                if (schemeSummaryModelToProcess.getAmountType().equals(AmountType.PERCENTAGE)) {
34705 amit.gupta 658
                    //Check for rejected CashDiscounts dont continue if its ever rejected once
659
                    if (SchemeType.CDS.contains(schemeSummaryModelToProcess.getSchemeType())) {
660
                        List<SchemeInOut> cdSchemeInOuts = schemeInOutRepository.selectAllByType(schemeSummaryModelToProcess.getSchemeType(), inventoryItem.getId());
661
                        LOGGER.info(cdSchemeInOuts);
662
                        if (!cdSchemeInOuts.isEmpty()) continue;
663
                    }
31903 amit.gupta 664
                    percentageToPay += schemeSummaryModelToProcess.getAmount();
665
                } else {
666
                    fixedToPay += schemeSummaryModelToProcess.getAmount();
667
                }
668
                payoutSchemeSummaryModelMap.put(schemeSummaryModelToProcess, new AmountModel(schemeSummaryModelToProcess.getAmount(), schemeSummaryModelToProcess.getAmountType()));
669
            }
670
        }
671
        double walletCredit = 0d;
672
        if (fixedToPay > 0) {
31913 amit.gupta 673
            double fixedRollout = fixedToPay * (100 / (100 + inventoryPayoutModel.getPercentageAmount()));
31903 amit.gupta 674
            for (Map.Entry<SchemeSummaryModel, AmountModel> schemeSummaryModelAmountModelEntry : payoutSchemeSummaryModelMap.entrySet()) {
675
                SchemeSummaryModel schemeSummaryModel = schemeSummaryModelAmountModelEntry.getKey();
676
                AmountModel amountModel = schemeSummaryModelAmountModelEntry.getValue();
677
                if (amountModel.getAmountType().equals(AmountType.FIXED)) {
678
                    SchemeInOut sio = new SchemeInOut();
679
                    sio.setSchemeId(schemeSummaryModel.getSchemeId());
680
                    sio.setInventoryItemId(inventoryItem.getId());
681
                    sio.setStatus(SchemePayoutStatus.CREDITED);
682
                    sio.setCreditTimestamp(LocalDateTime.now());
32155 amit.gupta 683
                    sio.setAmount((float) (fixedRollout * amountModel.getAmount() / fixedToPay) + schemeTypeCancelledAmountMap.getOrDefault(schemeSummaryModel.getSchemeType(), 0f));
34317 amit.gupta 684
                    if (schemeSummaryModel.getSchemeType().getTransactionType().equals(StockTransactionType.IN))
31903 amit.gupta 685
                        sio.setStatusDescription("Credited for GRN of IMEI-" + inventoryItem.getSerialNumber());
686
                    else
687
                        sio.setStatusDescription("Credited for Sale of IMEI-" + inventoryItem.getSerialNumber());
688
                    schemeInOutRepository.persist(sio);
689
                }
690
            }
691
            walletCredit += fixedRollout;
692
        }
693
 
694
        if (percentageToPay > 0) {
31987 amit.gupta 695
            LOGGER.info("inventoryPayoutModel.getFixedAmount() ----> {}", inventoryPayoutModel.getFixedAmount());
31914 amit.gupta 696
            double effectiveDP = inventoryPayoutModel.getDp() - (inventoryPayoutModel.getFixedAmount() + fixedToPay);
31903 amit.gupta 697
            double totalPercentage = inventoryPayoutModel.getPercentageAmount() + percentageToPay;
698
            double percentageRollout = effectiveDP * (totalPercentage / (100 + totalPercentage) - (inventoryPayoutModel.getPercentageAmount() / (100 + inventoryPayoutModel.getPercentageAmount())));
699
            for (Map.Entry<SchemeSummaryModel, AmountModel> schemeSummaryModelAmountModelEntry : payoutSchemeSummaryModelMap.entrySet()) {
700
                SchemeSummaryModel schemeSummaryModel = schemeSummaryModelAmountModelEntry.getKey();
701
                AmountModel amountModel = schemeSummaryModelAmountModelEntry.getValue();
702
                if (amountModel.getAmountType().equals(AmountType.PERCENTAGE)) {
703
                    SchemeInOut sio = new SchemeInOut();
704
                    sio.setInventoryItemId(inventoryItem.getId());
705
                    sio.setSchemeId(schemeSummaryModel.getSchemeId());
706
                    sio.setStatus(SchemePayoutStatus.CREDITED);
707
                    sio.setCreditTimestamp(LocalDateTime.now());
32155 amit.gupta 708
                    sio.setAmount((float) (percentageRollout * amountModel.getAmount() / percentageToPay) +
709
                            schemeTypeCancelledAmountMap.getOrDefault(schemeSummaryModel.getSchemeType(), 0f));
34317 amit.gupta 710
                    if (schemeSummaryModel.getSchemeType().getTransactionType().equals(StockTransactionType.IN))
31903 amit.gupta 711
                        sio.setStatusDescription("Credited for GRN of IMEI-" + inventoryItem.getSerialNumber());
712
                    else
713
                        sio.setStatusDescription("Credited for Sale of IMEI-" + inventoryItem.getSerialNumber());
714
                    schemeInOutRepository.persist(sio);
715
                }
716
            }
717
            walletCredit += percentageRollout;
718
        }
719
 
720
        return (float) walletCredit;
721
    }
722
 
723
    private Set<Integer> filterImeisByAgeing(Set<Integer> inventoryItemIds, FofoOrder fofoOrder) {
724
        Set<Integer> filteredInventoryIds = new HashSet<>();
725
        List<PartnerAgeingModel> partnerAgeingModels = ageingService.filterAgedInventory(inventoryItemIds);
726
        for (PartnerAgeingModel partnerAgeingModel : partnerAgeingModels) {
727
            LOGGER.info("Serial Number - {}", partnerAgeingModel.getSerialNumber());
728
            if (partnerAgeingModel.getBrand().equalsIgnoreCase("Samsung")) {
729
                ActivatedImei activatedImei = activatedImeiRepository.selectBySerialNumber(partnerAgeingModel.getSerialNumber());
730
                if (activatedImei != null && activatedImei.getActivationTimestamp().toLocalDate()
731
                        .isBefore(partnerAgeingModel.getBillingDate().plusDays(partnerAgeingModel.getMaxAgeingDays()))) {
732
                    //Lets give money if activation is there and is before the ageing limit of that brand.
733
                    filteredInventoryIds.add(partnerAgeingModel.getInventoryItemId());
734
                } else {
735
                    //If billing happens before ageing expiry
736
                    if (fofoOrder.getCreateTimestamp().toLocalDate().isBefore(partnerAgeingModel.getBillingDate().plusDays(partnerAgeingModel.getMaxAgeingDays()))) {
737
                        filteredInventoryIds.add(partnerAgeingModel.getInventoryItemId());
738
                    }
739
                }
740
            } else {
741
                filteredInventoryIds.add(partnerAgeingModel.getInventoryItemId());
742
            }
743
        }
744
        return filteredInventoryIds;
745
    }
746
 
33087 amit.gupta 747
    @Autowired
748
    WarehouseInventoryItemRepository warehouseInventoryItemRepository;
749
 
31410 amit.gupta 750
    @Override
751
    public float processSchemeOut(int fofoOrderId, int retailerId) throws ProfitMandiBusinessException {
34504 amit.gupta 752
        float totalCashback = 0;
31410 amit.gupta 753
        FofoOrder fofoOrder = fofoOrderRepository.selectByFofoIdAndOrderId(retailerId, fofoOrderId);
754
        // Process only if order is not cancelled
755
        if (fofoOrder.getCancelledTimestamp() == null) {
756
            // PartnerType partnerType = partnerTypeChangeService.getTypeOnDate(retailerId,
757
            // fofoOrder.getCreateTimestamp().toLocalDate());
758
            // TODO - SCHEME
759
            PartnerType partnerType = partnerTypeChangeService.getTypeOnMonth(retailerId,
760
                    YearMonth.from(fofoOrder.getCreateTimestamp()));
29927 amit.gupta 761
 
31410 amit.gupta 762
            List<ScanRecord> scanRecords = scanRecordRepository.selectAllByOrderId(fofoOrderId);
763
            if (scanRecords.size() == 0) return 0;
764
            Set<Integer> inventoryItemIds = scanRecords.stream().map(x -> x.getInventoryItemId())
765
                    .collect(Collectors.toSet());
31903 amit.gupta 766
            //Check for ageing
31912 amit.gupta 767
            //inventoryItemIds = this.filterImeisByAgeing(inventoryItemIds, fofoOrder);
31903 amit.gupta 768
            //ageingService.filterAgedInventory(inventoryItemIds);
31410 amit.gupta 769
            LOGGER.info("fofoOrderId --- {}", fofoOrderId);
770
            LOGGER.info("scanRecords --- {}", scanRecords);
771
            LOGGER.info("inventoryItemIds --- {}", inventoryItemIds);
31903 amit.gupta 772
 
773
            if (inventoryItemIds.size() == 0) return 0;
774
 
31410 amit.gupta 775
            Set<InventoryItem> inventoryItems = inventoryItemRepository.selectByIds(inventoryItemIds).stream()
776
                    .filter(x -> x.getSerialNumber() != null && !x.getSerialNumber().equals(""))
777
                    .collect(Collectors.toSet());
34221 tejus.loha 778
            inventoryItems = inventoryItems.stream().filter(inventoryItem -> !this.getBlockedImeis().contains(inventoryItem.getSerialNumber())).collect(Collectors.toSet());
29927 amit.gupta 779
 
33087 amit.gupta 780
            //Do not consider imei above 90 days for samsung
33432 amit.gupta 781
            List<String> samsungSerialNumbers = inventoryItems.stream().filter(x -> x.getItem().getBrand().equalsIgnoreCase("samsung")).map(x -> x.getSerialNumber()).collect(Collectors.toList());
33600 ranu 782
 
33432 amit.gupta 783
            if (samsungSerialNumbers.size() > 0) {
33087 amit.gupta 784
                List<AgeingSummaryModel> billedImeiModels = warehouseInventoryItemRepository.findStockAgeingByFofoIdSerialNumbers(retailerId, samsungSerialNumbers);
33606 ranu 785
                List<String> agedSerialNumbers = billedImeiModels.stream().filter(x -> x.isAgedAbove(365)).map(x -> x.getSerialNumber()).collect(Collectors.toList());
786
                if (agedSerialNumbers.size() > 0) {
787
                    List<String> samsungExceptionsSerialNumbers = samsungExceptionRepository.selectAllBySerialNumber(agedSerialNumbers).stream().map(x -> x.getSerialNumber()).collect(Collectors.toList());
788
                    agedSerialNumbers.removeAll(samsungExceptionsSerialNumbers);
789
                }
33432 amit.gupta 790
                inventoryItems = inventoryItems.stream().filter(x -> !agedSerialNumbers.contains(x.getSerialNumber())).collect(Collectors.toSet());
33087 amit.gupta 791
            }
792
 
31903 amit.gupta 793
            if (inventoryItems.size() == 0) return 0;
29927 amit.gupta 794
 
31903 amit.gupta 795
            Map<Integer, List<InventoryItem>> catalogInventoryItemMap = inventoryItems.stream().collect(Collectors.groupingBy(x -> x.getItem().getCatalogItemId()));
796
            Map<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSchemeSummaryMap = tagListingRepository.getModelSchemesByCatalogIdsAndType(retailerId,
797
                    partnerType, new ArrayList<>(catalogInventoryItemMap.keySet()), fofoOrder.getCreateTimestamp());
23444 amit.gupta 798
 
34317 amit.gupta 799
            LOGGER.info("catalogSchemeSummaryMap - {}", catalogSchemeSummaryMap);
31903 amit.gupta 800
 
34317 amit.gupta 801
 
31410 amit.gupta 802
            int count = 0;
31903 amit.gupta 803
            for (Map.Entry<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSummaryModelListEntry : catalogSchemeSummaryMap.entrySet()) {
804
                CatalogSummaryModel catalogSummaryModel = catalogSummaryModelListEntry.getKey();
32058 amit.gupta 805
                List<SchemeSummaryModel> schemeSummaryModels = catalogSummaryModelListEntry.getValue().stream().filter(x -> x != null).collect(Collectors.toList());
34317 amit.gupta 806
 
807
                schemeSummaryModels.stream().filter(x -> x.getSchemeType().getTransactionType().equals(StockTransactionType.OUT)).forEach(x -> x.setProcess(true));
31903 amit.gupta 808
                if (schemeSummaryModels.stream().filter(x -> x.isProcess()).count() == 0) continue;
23444 amit.gupta 809
 
31903 amit.gupta 810
                List<InventoryItem> modelInventoryItems = catalogInventoryItemMap.get(catalogSummaryModel.getCatalogId());
811
                for (InventoryItem inventoryItem : modelInventoryItems) {
34698 amit.gupta 812
                    Map<Integer, SchemeSummaryModel> schemeSummaryModelMap = schemeSummaryModels.stream().collect(Collectors.toMap(x -> x.getSchemeId(), x -> x));
34705 amit.gupta 813
                    List<SchemeInOut> sios = schemeInOutRepository.selectAll(inventoryItem.getId());
814
                    List<Integer> creditedSchemeIds = sios.stream()
815
                            .filter(x -> x.getStatus().equals(SchemePayoutStatus.CREDITED))
816
                            .map(x -> x.getSchemeId()).collect(Collectors.toList());
817
                    List<Integer> schemeIdsToReject = schemeRepository.selectBySchemeIds(creditedSchemeIds).stream().filter(x -> SchemeType.OUT_TYPES.contains(x.getType())).map(x -> x.getId()).collect(Collectors.toList());
34698 amit.gupta 818
 
819
                    //Reject invalid scheme payouts due to upgrade in Category or any change in schemes historically
34705 amit.gupta 820
                    for (SchemeInOut sio : sios) {
821
                        if (schemeIdsToReject.contains(sio.getSchemeId())) {
822
                            sio.setStatusDescription("Rolledback due to Category upgrade/invalid scheme");
823
                            sio.setStatus(SchemePayoutStatus.REJECTED);
824
                            sio.setRolledBackTimestamp(LocalDateTime.now());
825
                            totalCashback -= sio.getAmount();
826
                        }
34698 amit.gupta 827
                    }
32165 amit.gupta 828
                    float inventoryItemCashback = this.createSchemeInOut(schemeSummaryModels, inventoryItem);
31903 amit.gupta 829
                    if (inventoryItemCashback > 0) {
830
                        count++;
831
                        totalCashback += inventoryItemCashback;
832
                    }
31410 amit.gupta 833
                }
834
            }
31903 amit.gupta 835
 
31410 amit.gupta 836
            if (count > 0) {
837
                walletService.addAmountToWallet(
838
                        retailerId, fofoOrderId, WalletReferenceType.SCHEME_OUT, "Sales margin for invoice number "
839
                                + fofoOrder.getInvoiceNumber() + ". Total " + count + " pc(s)",
840
                        totalCashback, fofoOrder.getCreateTimestamp());
841
                fofoOrder.setCashback(totalCashback + fofoOrder.getCashback());
842
            }
843
        }
34504 amit.gupta 844
        return totalCashback;
31410 amit.gupta 845
    }
22653 ashik.ali 846
 
31410 amit.gupta 847
    @Override
848
    //Tax rate has been passed to 0 to ensure no tax deduction
849
    public float getSpecialSupportAmount(float supportAmount, PartnerType partnerType, LocalDate onDate,
850
                                         int catalogId) throws ProfitMandiBusinessException {
851
        //int itemId = itemRepository.selectAllByCatalogItemId(catalogId).stream().findAny().get().getId();
852
        //float totalTaxRate = stateGstRateRepository.getTotalTaxRate(itemId);
853
        return this.getSpecialSupportAmount(supportAmount, partnerType, onDate, catalogId, 0);
854
    }
23444 amit.gupta 855
 
31410 amit.gupta 856
    @Override
857
    public float getSpecialSupportAmount(float supportAmount, PartnerType partnerType, LocalDate onDate,
858
                                         int catalogId, float taxRate) throws ProfitMandiBusinessException {
859
        float totalMargin = this.selectPercentageScheme(partnerType, onDate, catalogId, false, 0, 0).stream().collect(Collectors.summingDouble(x -> x.getAmount())).floatValue();
860
        float amountToCredit = supportAmount * (1 - (totalMargin / (100 + taxRate)));
861
        return amountToCredit;
862
    }
23444 amit.gupta 863
 
31410 amit.gupta 864
    @Override
33614 amit.gupta 865
    public void rollbackSchemes(List<Integer> inventoryItemIds, String rollbackReason)
31410 amit.gupta 866
            throws Exception {
33614 amit.gupta 867
        List<InventoryItem> inventoryItems = inventoryItemRepository.selectAllByIds(inventoryItemIds);
868
        Map<Integer, InventoryItem> inventoryItemMap = inventoryItems.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
869
        Map<Integer, Integer> purchasePartnerMap = inventoryItems.stream().collect(Collectors.toMap(x -> x.getPurchaseId(), x -> x.getFofoId(), (u, v) -> u));
870
 
871
        LOGGER.info("inventoryItemIds - {}", inventoryItemIds);
872
        List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(new HashSet<>(inventoryItemIds));
873
        List<Integer> schemeIds = schemeInOuts.stream().map(x -> x.getSchemeId()).distinct().collect(Collectors.toList());
34139 amit.gupta 874
        if (schemeIds.size() == 0) return;
33614 amit.gupta 875
        List<Scheme> schemes = schemeRepository.selectBySchemeIds(schemeIds);
876
        Map<Integer, Float> inSchemesMap = new HashMap<>();
877
        Map<Integer, Float> outSchemesMap = new HashMap<>();
878
        Map<Integer, Scheme> schemesMap = schemes.stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
879
        for (SchemeInOut schemeInOut : schemeInOuts) {
880
            Map<Integer, Float> schemePayoutMap;
881
            int inventoryItemId = schemeInOut.getInventoryItemId();
31410 amit.gupta 882
            if (schemeInOut.getRolledBackTimestamp() == null) {
883
                schemeInOut.setRolledBackTimestamp(LocalDateTime.now());
884
                if (schemeInOut.getStatus() == null || schemeInOut.getStatus().equals(SchemePayoutStatus.CREDITED)) {
33614 amit.gupta 885
                    Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
34317 amit.gupta 886
                    if (scheme.getType().getTransactionType().equals(StockTransactionType.IN)) {
33614 amit.gupta 887
                        schemePayoutMap = inSchemesMap;
888
                    } else {
889
                        schemePayoutMap = outSchemesMap;
890
                    }
891
                    if (!schemePayoutMap.containsKey(inventoryItemId)) {
892
                        schemePayoutMap.put(inventoryItemId, 0f);
893
                    }
34139 amit.gupta 894
                    schemePayoutMap.put(inventoryItemId, inSchemesMap.get(inventoryItemId) == null ? 0 : inSchemesMap.get(inventoryItemId) + schemeInOut.getAmount());
31410 amit.gupta 895
                }
896
                schemeInOut.setStatus(SchemePayoutStatus.REJECTED);
897
                schemeInOut.setStatusDescription(rollbackReason);
898
            }
899
        }
33614 amit.gupta 900
        Map<Integer, Double> purchaseRollbackAmountMap = inSchemesMap.entrySet().stream().collect(Collectors.groupingBy(x -> inventoryItemMap.get(x.getKey()).getPurchaseId(), Collectors.summingDouble(x -> x.getValue())));
901
 
902
        for (Map.Entry<Integer, Double> purchaseRollbackAmountEntry : purchaseRollbackAmountMap.entrySet()) {
903
            int purchaseId = purchaseRollbackAmountEntry.getKey();
904
            Double amountToRollback = purchaseRollbackAmountEntry.getValue();
905
            if (amountToRollback != null && amountToRollback > 0) {
906
                walletService.rollbackAmountFromWallet(purchasePartnerMap.get(purchaseId), amountToRollback.floatValue(), purchaseId,
907
                        WalletReferenceType.SCHEME_IN, rollbackReason, LocalDateTime.now());
908
            }
31410 amit.gupta 909
        }
910
    }
23444 amit.gupta 911
 
31410 amit.gupta 912
    @Override
913
    public Map<String, Object> getSchemes(Set<Integer> roleIds, int offset, int limit)
914
            throws ProfitMandiBusinessException {
915
        Map<String, Object> map = new HashMap<>();
916
        List<Scheme> schemes = null;
917
        long size = 0;
918
        if (roleManager.isAdmin(roleIds)) {
919
            schemes = schemeRepository.selectAll(offset, limit);
920
            size = schemeRepository.selectAllCount();
921
        } else {
922
            schemes = schemeRepository.selectActiveAll(offset, limit);
923
            size = schemeRepository.selectAllActiveCount();
924
        }
925
        map.put("schemes", schemes);
926
        map.put("start", offset + 1);
927
        map.put("size", size);
928
        if (schemes.size() < limit) {
929
            map.put("end", offset + schemes.size());
930
        } else {
931
            map.put("end", offset + limit);
932
        }
933
        return map;
934
    }
26722 amit.gupta 935
 
31410 amit.gupta 936
    @Override
937
    public List<Scheme> getPaginatedSchemes(Set<Integer> roleIds, int offset, int limit)
938
            throws ProfitMandiBusinessException {
939
        LOGGER.info("requested offset=[{}], limit = [{}]", offset, limit);
940
        List<Scheme> schemes = null;
941
        if (roleManager.isAdmin(roleIds)) {
942
            schemes = schemeRepository.selectAll(offset, limit);
943
        } else {
944
            schemes = schemeRepository.selectActiveAll(offset, limit);
945
        }
946
        return schemes;
947
    }
27898 amit.gupta 948
 
31410 amit.gupta 949
    @Override
950
    // This is being called to reverse schemes while processing price Drop
951
    public void reverseSchemes(List<InventoryItem> inventoryItems, int priceDropId, String reversalReason)
952
            throws ProfitMandiBusinessException {
953
        PriceDrop priceDrop = priceDropRepository.selectById(priceDropId);
954
        Map<Integer, List<InventoryItem>> purchaseInventoryListMap = inventoryItems.stream()
955
                .collect(Collectors.groupingBy(InventoryItem::getPurchaseId, Collectors.toList()));
29927 amit.gupta 956
 
31410 amit.gupta 957
        for (Map.Entry<Integer, List<InventoryItem>> purchaseEntry : purchaseInventoryListMap.entrySet()) {
958
            float amountToCredit = 0;
959
            float amountToDebit = 0;
960
            int purchaseId = purchaseEntry.getKey();
961
            List<InventoryItem> purchaseInventoryItemList = purchaseEntry.getValue();
30454 amit.gupta 962
 
31410 amit.gupta 963
            Map<Integer, InventoryItem> inventoryItemsMap = purchaseInventoryItemList.stream()
964
                    .collect(Collectors.toMap(x -> x.getId(), x -> x));
29927 amit.gupta 965
 
31410 amit.gupta 966
            List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(inventoryItemsMap.keySet());
967
            LOGGER.info("Scheme InOuts , {}", schemeInOuts);
968
            if (schemeInOuts.size() == 0) {
969
                continue;
970
            }
971
            List<Integer> schemeIds = schemeInOuts.stream().map(x -> x.getSchemeId()).collect(Collectors.toList());
972
            Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds, 0, schemeIds.size())
973
                    .stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
974
            for (SchemeInOut schemeInOut : schemeInOuts) {
975
                InventoryItem ii = inventoryItemsMap.get(schemeInOut.getInventoryItemId());
976
                Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
977
                if (scheme.getAmountType().equals(AmountType.FIXED)) {
978
                    continue;
979
                }
34317 amit.gupta 980
                if (scheme.getType().getTransactionType().equals(StockTransactionType.IN) && schemeInOut.getRolledBackTimestamp() == null) {
33432 amit.gupta 981
                    schemeInOut.setRolledBackTimestamp(LocalDateTime.now());
982
                    schemeInOut.setStatus(SchemePayoutStatus.REJECTED);
983
                    schemeInOut.setStatusDescription("Margin reversed due to price drop");
984
                    // IF not credited then dont consider any credit/debit for that sio entry
985
                    if (schemeInOut.getCreditTimestamp() != null) {
986
                        amountToDebit += schemeInOut.getAmount();
31410 amit.gupta 987
                    }
988
                }
989
            }
990
            int fofoId = inventoryItems.get(0).getFofoId();
991
            if (amountToDebit > 0) {
992
                walletService.addAmountToWallet(fofoId, purchaseId, WalletReferenceType.SCHEME_IN,
993
                        MessageFormat.format(reversalReason, purchaseInventoryItemList.size()), -amountToDebit,
994
                        priceDrop.getAffectedOn());
995
            }
996
        }
997
    }
29231 amit.gupta 998
 
31609 amit.gupta 999
    @Autowired
32060 amit.gupta 1000
    UserWalletRepository userWalletRepository;
1001
    @Autowired
31609 amit.gupta 1002
    UserWalletHistoryRepository userWalletHistoryRepository;
1003
 
31410 amit.gupta 1004
    @Override
1005
    // Always being called from cancel order/bad return means no SCHEME IN is considered
1006
    public void reverseSchemes(List<InventoryItem> inventoryItems, int reversalReference, String reversalReason,
1007
                               List<SchemeType> schemeTypes) throws ProfitMandiBusinessException {
1008
        Map<Integer, InventoryItem> inventoryItemsMap = inventoryItems.stream()
1009
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1010
        LOGGER.info("inventoryItems" + inventoryItems);
23444 amit.gupta 1011
 
31410 amit.gupta 1012
        Map<SchemeType, SchemeInOut> schemeTypeMap = new HashMap<>();
30454 amit.gupta 1013
 
31410 amit.gupta 1014
        List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByInventoryItemIds(inventoryItemsMap.keySet());
32060 amit.gupta 1015
        List<SchemeInOut> rolledBacks = schemeInOuts.stream().filter(x -> x.getStatusDescription().equals(reversalReason)).collect(Collectors.toList());
31410 amit.gupta 1016
        float amountToRollback = 0;
23444 amit.gupta 1017
 
31410 amit.gupta 1018
        if (!schemeInOuts.isEmpty()) {
1019
            List<Integer> schemeIds = schemeInOuts.stream().map(x -> x.getSchemeId()).collect(Collectors.toList());
1020
            LOGGER.info("schemeIds" + schemeIds);
29231 amit.gupta 1021
 
31410 amit.gupta 1022
            Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds, 0, schemeIds.size())
1023
                    .stream().collect(Collectors.toMap(x -> x.getId(), x -> x));
32060 amit.gupta 1024
            if (rolledBacks.size() > 0) {
1025
                for (SchemeInOut schemeInOut : rolledBacks) {
1026
                    Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
1027
                    if (schemeTypes.contains(scheme.getType())) {
1028
                        schemeTypeMap.put(scheme.getType(), schemeInOut);
1029
                        if (schemeInOut.getCreditTimestamp() != null) {
31410 amit.gupta 1030
                            amountToRollback += schemeInOut.getAmount();
1031
                        }
1032
                    }
1033
                }
32060 amit.gupta 1034
            } else {
1035
                for (SchemeInOut schemeInOut : schemeInOuts) {
1036
                    Scheme scheme = schemesMap.get(schemeInOut.getSchemeId());
1037
                    if (schemeTypes.contains(scheme.getType())) {
1038
                        if (schemeInOut.getRolledBackTimestamp() == null) {
1039
                            schemeInOut.setRolledBackTimestamp(LocalDateTime.now());
1040
                            if (schemeInOut.getStatus().equals(SchemePayoutStatus.CREDITED)) {
1041
                                amountToRollback += schemeInOut.getAmount();
1042
                            }
1043
                            schemeInOut.setStatus(SchemePayoutStatus.REJECTED);
1044
                            schemeInOut.setStatusDescription(reversalReason);
1045
                        }
1046
                    }
1047
                }
31410 amit.gupta 1048
            }
1049
        }
32060 amit.gupta 1050
 
31410 amit.gupta 1051
        int fofoId = inventoryItems.get(0).getFofoId();
34486 amit.gupta 1052
        WalletReferenceType walletReferenceType = schemeTypes.containsAll(SchemeType.OUT_SCHEME_TYPES) ? WalletReferenceType.SCHEME_OUT
31609 amit.gupta 1053
                : schemeTypes.contains(SchemeType.SPECIAL_SUPPORT) ? WalletReferenceType.SPECIAL_SUPPORT
34486 amit.gupta 1054
                : schemeTypes.contains(SchemeType.INVESTMENT) ? WalletReferenceType.SCHEME_OUT : null;
32060 amit.gupta 1055
        List<UserWalletHistory> userWalletHistoryList = null;
32058 amit.gupta 1056
        if (amountToRollback > 0 && walletReferenceType != null) {
1057
            // Mark appropriate reference of rollback investment margin
32060 amit.gupta 1058
            if (schemeTypes.contains(SchemeType.INVESTMENT) && schemeTypeMap.containsKey(SchemeType.INVESTMENT)) {
1059
                LocalDateTime creditTime = schemeTypeMap.get(SchemeType.INVESTMENT).getCreditTimestamp();
1060
                if (creditTime == null) return;
1061
                int investmentReversalReference = Integer.parseInt(FormattingUtils.getYearMonth(creditTime.minusMonths(1)));
1062
                UserWallet userWallet = userWalletRepository.selectByRetailerId(fofoId);
1063
                userWalletHistoryList = userWalletHistoryRepository.selectAllByreferenceIdandreferenceType(userWallet.getUserId(), investmentReversalReference, WalletReferenceType.INVESTMENT_PAYOUT);
1064
                if (userWalletHistoryList.size() > 0) {
1065
                    walletReferenceType = WalletReferenceType.INVESTMENT_PAYOUT;
1066
                } else {
1067
                    userWalletHistoryList = null;
32058 amit.gupta 1068
                }
32060 amit.gupta 1069
            }
1070
            if (userWalletHistoryList == null) {
1071
                userWalletHistoryList = userWalletHistoryRepository.selectAllByreferenceIdandreferenceType(reversalReference, walletReferenceType);
1072
            }
1073
            if (userWalletHistoryList.size() > 0) {
1074
                int maxDeductible = userWalletHistoryList.stream().collect(Collectors.summingInt(x -> x.getAmount()));
1075
                if (maxDeductible > 0) {
32165 amit.gupta 1076
                    LOGGER.info("----------> maxDeductible {}, amountToRollback {}, reversalReference {}, walletReferenceType {} ", maxDeductible, amountToRollback, reversalReference, walletReferenceType);
32060 amit.gupta 1077
                    walletService.rollbackAmountFromWallet(fofoId, Math.min(maxDeductible, amountToRollback), reversalReference, walletReferenceType,
1078
                            reversalReason, LocalDateTime.now());
1079
                }
31609 amit.gupta 1080
 
31903 amit.gupta 1081
            }
32058 amit.gupta 1082
        }
31410 amit.gupta 1083
    }
29231 amit.gupta 1084
 
32060 amit.gupta 1085
 
31410 amit.gupta 1086
    @Override
1087
    public double getTotalMargin(int itemId, PartnerType partnerType, LocalDateTime dateTime) {
1088
        Session session = sessionFactory.getCurrentSession();
1089
        CriteriaBuilder cb = session.getCriteriaBuilder();
1090
        CriteriaQuery<Double> criteriaQuery = cb.createQuery(Double.class);
1091
        Root<SchemeItem> schemeItem = criteriaQuery.from(SchemeItem.class);
1092
        Root<Scheme> scheme = criteriaQuery.from(Scheme.class);
1093
        Predicate schemePredicate = cb.equal(scheme.get(ProfitMandiConstants.AMOUNT_TYPE), AmountType.PERCENTAGE);
1094
        Predicate lessThanPredicate = cb.lessThanOrEqualTo(scheme.get(ProfitMandiConstants.END_DATE_TIME), dateTime);
1095
        Predicate greaterThanPredicate = cb.greaterThanOrEqualTo(scheme.get(ProfitMandiConstants.START_DATE_TIME),
1096
                dateTime);
1097
        Predicate joinPredicate = cb.equal(scheme.get("id"), schemeItem.get("schemeId"));
1098
        Predicate schemeItemPredicate = cb.equal(schemeItem.get(ProfitMandiConstants.ITEM_ID), itemId);
1099
        criteriaQuery.select(cb.sum(scheme.get(ProfitMandiConstants.AMOUNT))).where(schemePredicate, lessThanPredicate,
1100
                greaterThanPredicate, schemeItemPredicate, joinPredicate);
29231 amit.gupta 1101
 
31410 amit.gupta 1102
        Query<Double> query = session.createQuery(criteriaQuery);
1103
        return query.getSingleResult() + ProfitMandiConstants.SCHEME_INVESTMENT_MARGIN;
29231 amit.gupta 1104
 
31410 amit.gupta 1105
    }
23444 amit.gupta 1106
 
32972 amit.gupta 1107
 
31410 amit.gupta 1108
    @Override
33614 amit.gupta 1109
    public Map<Integer, Float> getCatalogSchemeCashBack(int fofoId, List<Integer> catalogIds) throws
1110
            ProfitMandiBusinessException {
32972 amit.gupta 1111
        PartnerType partnerType = partnerTypeChangeService.getTypeOnDate(fofoId, LocalDate.now());
1112
        Map<CatalogSummaryModel, List<SchemeSummaryModel>> catalogModelMap = tagListingRepository.getModelSchemesByCatalogIdsAndType(fofoId, partnerType, catalogIds, LocalDate.now().atStartOfDay());
1113
 
1114
        Map<Integer, Float> catalogCashbackMap = new HashMap<>();
1115
        for (Map.Entry<CatalogSummaryModel, List<SchemeSummaryModel>> catalogSummaryModelListEntry : catalogModelMap.entrySet()) {
1116
            int catalogItemId = catalogSummaryModelListEntry.getKey().getCatalogId();
1117
            List<SchemeSummaryModel> schemeSummaryModels = catalogSummaryModelListEntry.getValue();
1118
 
1119
            float totalCashback = schemeSummaryModels.stream()
34317 amit.gupta 1120
                    .filter(x -> Arrays.asList(SchemeType.SPECIAL_SUPPORT, SchemeType.SELLOUT).contains(x.getSchemeType())
32972 amit.gupta 1121
                            && x.getAmountType().equals(AmountType.FIXED))
1122
                    .collect(Collectors.summingDouble(x -> x.getAmount())).floatValue();
1123
            catalogCashbackMap.put(catalogItemId, totalCashback);
31410 amit.gupta 1124
        }
32972 amit.gupta 1125
        return catalogCashbackMap;
31410 amit.gupta 1126
    }
30572 amit.gupta 1127
 
31410 amit.gupta 1128
    @Override
31903 amit.gupta 1129
    public List<Scheme> selectSchemeByPartnerTypeFofoId(PartnerType partnerType, LocalDate onDate, int catalogId,
1130
                                                        int fofoId, int offset, int limit) throws ProfitMandiBusinessException {
31410 amit.gupta 1131
        Session session = sessionFactory.getCurrentSession();
1132
        final TypedQuery<Scheme> typedQuery = session.createNamedQuery(
1133
                "Scheme.selectSchemeByModelsPartnerTypeFofoId", Scheme.class);
1134
        typedQuery.setParameter("catalogIds", Arrays.asList(catalogId));
1135
        typedQuery.setParameter("fofoIds", Arrays.asList(fofoId, 0));
1136
        typedQuery.setParameter("onDate", onDate.atStartOfDay());
1137
        typedQuery.setParameter("partnerTypes", Arrays.asList(partnerType, partnerType.ALL));
1138
        typedQuery.setFirstResult(offset);
1139
        if (limit != 0) {
1140
            typedQuery.setMaxResults(limit);
1141
        }
1142
        return typedQuery.getResultList();
1143
    }
30463 amit.gupta 1144
 
33432 amit.gupta 1145
    @Override
1146
    public void processSchemeIn(List<InventoryItem> inventoryItems) throws ProfitMandiBusinessException {
1147
        Map<Integer, List<InventoryItem>> purchaseIdInventoryItemsMap = inventoryItems.stream().collect(Collectors.groupingBy(x -> x.getPurchaseId()));
1148
        for (Map.Entry<Integer, List<InventoryItem>> purchaseIdInventoryItemEntry : purchaseIdInventoryItemsMap.entrySet()) {
34317 amit.gupta 1149
            int retailerId = purchaseIdInventoryItemEntry.getValue().get(0).getFofoId();
1150
            this.processSchemeIn(purchaseIdInventoryItemEntry.getKey(), retailerId);
33432 amit.gupta 1151
        }
1152
    }
1153
 
34698 amit.gupta 1154
    /*@Override
1155
    public void processFullCD(String invoiceNumber) throws ProfitMandiBusinessException {
1156
        Purchase purchase = purchaseRepository.selectByPurchaseReference(invoiceNumber);
1157
        this.processSchemeIn(purchase.getId(), purchase.getFofoId(), SchemeType.CDS);
1158
    }
1159
 
1160
    @Override
1161
    public void processPartialCD(String invoiceNumber) throws ProfitMandiBusinessException {
1162
        Purchase purchase = purchaseRepository.selectByPurchaseReference(invoiceNumber);
1163
        this.processSchemeIn(purchase.getId(), purchase.getFofoId(), Arrays.asList(SchemeType.CASH_DISCOUNT));
1164
    }
1165
 
1166
    @Override
1167
    public void processNoCD(String invoiceNumber) throws ProfitMandiBusinessException {
1168
        Purchase purchase = purchaseRepository.selectByPurchaseReference(invoiceNumber);
1169
        this.processSchemeIn(purchase.getId(), purchase.getFofoId(), Arrays.asList());
1170
    }*/
1171
 
32309 amit.gupta 1172
    @Autowired
1173
    FofoStoreRepository fofoStoreRepository;
1174
 
1175
 
31410 amit.gupta 1176
    @Override
32309 amit.gupta 1177
    @Cacheable(value = "staticscheme", cacheManager = "oneDayCacheManager")
1178
    public Scheme getStaticScheme(int fofoId) throws ProfitMandiBusinessException {
33165 amit.gupta 1179
        FofoStore fofoStore = fofoStoreRepository.selectByRetailerId(fofoId);
32504 amit.gupta 1180
        Scheme scheme = null;
1181
        if (fofoStore.getTarget() > 0) {
1182
            scheme = new Scheme();
33704 amit.gupta 1183
            scheme.setName("Super Retailer - Club 3");
1184
            scheme.setStartDateTime(LocalDate.of(2024, 8, 1).atStartOfDay());
1185
            scheme.setEndDateTime(LocalDate.of(2024, 11, 30).atTime(LocalTime.MAX));
32504 amit.gupta 1186
            scheme.setTarget(fofoStore.getTarget());
33548 amit.gupta 1187
 
33704 amit.gupta 1188
            if (scheme.getEndDateTime().plusDays(5).isBefore(LocalDateTime.now())) return null;
32504 amit.gupta 1189
        }
33165 amit.gupta 1190
        return scheme;
32309 amit.gupta 1191
    }
1192
 
1193
    @Override
31410 amit.gupta 1194
    public List<Scheme> selectSchemeByPartnerType(PartnerType partnerType, LocalDate onDate, int catalogId,
1195
                                                  boolean isAdmin, int offset, int limit) throws ProfitMandiBusinessException {
1196
        Session session = sessionFactory.getCurrentSession();
1197
        List<Predicate> andPredicates = new ArrayList<>();
1198
        CriteriaBuilder cb = session.getCriteriaBuilder();
1199
        CriteriaQuery<Scheme> query = cb.createQuery(Scheme.class);
1200
        Root<Scheme> scheme = query.from(Scheme.class);
1201
        if (!partnerType.equals(PartnerType.ALL)) {
1202
            List<PartnerType> pt = new ArrayList<>();
1203
            pt.add(PartnerType.ALL);
1204
            pt.add(partnerType);
1205
            andPredicates.add(cb.in(scheme.get("partnerType")).value(pt));
1206
        }
1207
        cb.desc(cb.isNull(scheme.get("expireTimestamp")));
1208
        if (catalogId > 0) {
30454 amit.gupta 1209
 
31410 amit.gupta 1210
            List<Integer> schemeIds = schemeItemRepository.selectSchemeIdByCatalogId(catalogId);
1211
            LOGGER.info("schemeId" + schemeIds);
1212
            if (schemeIds.isEmpty()) {
1213
                return new ArrayList<>();
1214
            }
1215
            andPredicates.add(cb.in(scheme.get("id")).value(schemeIds));
1216
            if (onDate != null) {
1217
                andPredicates.add(cb.greaterThan(scheme.get("endDateTime"), onDate.atStartOfDay()));
1218
                andPredicates.add(cb.lessThanOrEqualTo(scheme.get("startDateTime"), onDate.atStartOfDay()));
1219
            }
1220
        }
1221
        if (!isAdmin) {
1222
            andPredicates.add(cb.isNotNull(scheme.get("activeTimestamp")));
1223
        }
1224
        query.where(cb.and(andPredicates.toArray(new Predicate[0])));
1225
        query.orderBy(cb.desc(cb.function("isnull", Boolean.class, scheme.get("expireTimestamp"))));
1226
        if (limit == 0) {
1227
            return session.createQuery(query).setFirstResult(offset).getResultList();
1228
        }
1229
        return session.createQuery(query).setFirstResult(offset).setMaxResults(limit).getResultList();
30454 amit.gupta 1230
 
31410 amit.gupta 1231
    }
30454 amit.gupta 1232
 
31410 amit.gupta 1233
    @Override
1234
    public List<Scheme> selectPercentageScheme(PartnerType partnerType, LocalDate onDate, int catalogId,
1235
                                               boolean isAdmin, int offset, int limit) throws ProfitMandiBusinessException {
1236
        List<Scheme> schemes = this.selectSchemeByPartnerType(partnerType, onDate, catalogId, isAdmin, offset, limit);
1237
        return schemes.stream().filter(x -> x.getAmountType().equals(AmountType.PERCENTAGE)).collect(Collectors.toList());
1238
    }
30454 amit.gupta 1239
 
31410 amit.gupta 1240
    @Override
1241
    public void processActivation() throws ProfitMandiBusinessException {
1242
        List<SchemeInOut> pendingPayouts = schemeInOutRepository.selectAllPending();
1243
        List<Integer> schemeIds = new ArrayList<>();
1244
        Set<Integer> inventoryIds = new HashSet<>();
1245
        for (SchemeInOut pendingPayout : pendingPayouts) {
1246
            schemeIds.add(pendingPayout.getSchemeId());
1247
        }
1248
        Map<Integer, Scheme> schemesMap = schemeRepository.selectBySchemeIds(schemeIds, 0, 0).stream()
34317 amit.gupta 1249
                .filter(x -> x.getType().equals(SchemeType.SPECIAL_SUPPORT))
31410 amit.gupta 1250
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1251
        pendingPayouts = pendingPayouts.stream().filter(x -> schemesMap.get(x.getSchemeId()) != null)
1252
                .collect(Collectors.toList());
23884 amit.gupta 1253
 
31410 amit.gupta 1254
        for (SchemeInOut pendingPayout : pendingPayouts) {
1255
            inventoryIds.add(pendingPayout.getInventoryItemId());
1256
        }
1257
        Map<Integer, InventoryItem> inventoryItemMap = inventoryItemRepository.selectByIds(inventoryIds).stream()
1258
                .collect(Collectors.toMap(x -> x.getId(), x -> x));
1259
        Map<String, InventoryItem> serialNumberMap = inventoryItemMap.values().stream()
1260
                .collect(Collectors.toMap(x -> x.getSerialNumber(), x -> x));
23796 amit.gupta 1261
 
31410 amit.gupta 1262
        List<ActivatedImei> activatedImeis = activatedImeiRepository
1263
                .selectBySerialNumbers(new ArrayList<>(serialNumberMap.keySet())).stream().collect(Collectors.toList());
23508 amit.gupta 1264
 
31410 amit.gupta 1265
        Map<String, ActivatedImei> activatedImeiMap = activatedImeis.stream()
1266
                .collect(Collectors.toMap(x -> x.getSerialNumber().toLowerCase(), x -> x));
1267
        for (SchemeInOut pendingPayout : pendingPayouts) {
31914 amit.gupta 1268
            Scheme scheme = schemesMap.get(pendingPayout.getSchemeId());
31410 amit.gupta 1269
            InventoryItem ii = inventoryItemMap.get(pendingPayout.getInventoryItemId());
1270
            String serialNumber = ii.getSerialNumber().toLowerCase();
1271
            ActivatedImei activatedImei = activatedImeiMap.get(serialNumber);
1272
            if (activatedImei == null) {
1273
                continue;
1274
            }
1275
            if (scheme.isWithinRange(activatedImei.getActivationTimestamp())) {
1276
                int fofoId = ii.getFofoId();
1277
                // Get latest order Id
1278
                int orderId = scanRecordRepository.selectByInventoryItemId(ii.getId()).stream()
1279
                        .filter(x -> x.getOrderId() > 0)
1280
                        .sorted(Comparator.comparing(ScanRecord::getCreateTimestamp).reversed()).findFirst().get()
1281
                        .getOrderId();
1282
                FofoOrder fofoOrder = fofoOrderRepository.selectByOrderId(orderId);
31916 amit.gupta 1283
 
31921 amit.gupta 1284
                InventoryPayoutModel inventoryPayoutModel = priceCircularService.getPayouts(ii);
31916 amit.gupta 1285
                AmountModel amountModel = new AmountModel();
1286
                amountModel.setAmount(scheme.getAmount());
1287
                amountModel.setAmountType(scheme.getAmountType());
31921 amit.gupta 1288
                double amountToRollout = inventoryPayoutModel.getRolloutAmount(amountModel);
32058 amit.gupta 1289
                pendingPayout.setAmount((float) amountToRollout);
34317 amit.gupta 1290
                walletService.addAmountToWallet(fofoId, orderId, WalletReferenceType.SPECIAL_SUPPORT,
1291
                        "Special Support for " + ii.getItem().getItemDescriptionNoColor() + ", Imei - " + serialNumber, (float) amountToRollout,
1292
                        fofoOrder.getCreateTimestamp());
1293
                pendingPayout.setStatusDescription("Special support credited, activated on " + FormattingUtils.formatDate(activatedImei.getActivationTimestamp()));
1294
 
31410 amit.gupta 1295
                pendingPayout.setCreditTimestamp(LocalDateTime.now());
1296
                pendingPayout.setStatus(SchemePayoutStatus.CREDITED);
1297
            } else {
1298
                pendingPayout.setStatus(SchemePayoutStatus.REJECTED);
1299
                pendingPayout.setRolledBackTimestamp(LocalDateTime.now());
1300
                pendingPayout.setStatusDescription(
1301
                        "Rejected, activated on " + FormattingUtils.formatDate(activatedImei.getActivationTimestamp()));
1302
            }
1303
        }
1304
    }
24976 amit.gupta 1305
 
31410 amit.gupta 1306
    @Override
1307
    public void processActivatedImeisForSchemes() throws ProfitMandiBusinessException {
1308
        List<SchemesImeisModel> schemesImeisModels = schemeRepository.selectSelectUnpaidSchemes();
1309
        LOGGER.info("Total Size - " + schemesImeisModels.size());
1310
        List<Integer> orderIds = schemesImeisModels.stream().map(x -> x.getOrderId()).collect(Collectors.toList());
1311
        List<FofoOrder> fofoOrders = fofoOrderRepository.selectAllByOrderIds(orderIds);
1312
        Map<Integer, FofoOrder> validOrdersMap = fofoOrders.stream().filter(x -> x.getCancelledTimestamp() == null).collect(Collectors.toMap(x -> x.getId(), x -> x));
1313
        Map<String, List<SchemesImeisModel>> validImeiSchemesModelMap = schemesImeisModels.stream().filter(x -> validOrdersMap.containsKey(x.getOrderId())).collect(Collectors.groupingBy(x -> x.getImei()));
1314
        for (Map.Entry<String, List<SchemesImeisModel>> imeiListEntry : validImeiSchemesModelMap.entrySet()) {
1315
            SchemesImeisModel schemesImeisModel = imeiListEntry.getValue().get(0);
1316
            List<Integer> schemeIds = imeiListEntry.getValue().stream().map(x -> x.getSchemeId()).collect(Collectors.toList());
1317
            LOGGER.info("Serial Number  - {}, Scheme IDs - {}", schemesImeisModel.getImei(), schemeIds);
1318
            InventoryItem inventoryItem = inventoryItemRepository.selectById(schemesImeisModel.getInventoryItemId());
1319
            List<Scheme> schemes = schemeRepository.selectBySchemeIds(schemeIds);
34317 amit.gupta 1320
            List<Scheme> supportSchemes = schemes.stream().filter(x -> Arrays.asList(SchemeType.SPECIAL_SUPPORT).contains(x.getType())).collect(Collectors.toList());
31410 amit.gupta 1321
            if (supportSchemes.size() > 0) {
32165 amit.gupta 1322
                for (Scheme scheme : supportSchemes) {
1323
                    List<SchemeInOut> schemeInOuts = schemeInOutRepository.selectByScheme(scheme.getId(), inventoryItem.getId());
32972 amit.gupta 1324
                    if (schemeInOuts.stream().filter(x -> Arrays.asList(SchemePayoutStatus.CREDITED, SchemePayoutStatus.PENDING).contains(x.getStatus())).count() > 0) {
32165 amit.gupta 1325
                        continue;
1326
                    }
1327
                    SchemeInOut sio = new SchemeInOut();
1328
                    sio.setAmount(0);
1329
                    sio.setInventoryItemId(inventoryItem.getId());
1330
                    sio.setSchemeId(scheme.getId());
1331
                    sio.setStatusDescription("Activation pending for IMEI#" + inventoryItem.getSerialNumber());
1332
                    sio.setStatus(SchemePayoutStatus.PENDING);
1333
                    schemeInOutRepository.persist(sio);
1334
                }
31410 amit.gupta 1335
            }
1336
        }
1337
    }
22653 ashik.ali 1338
}