Subversion Repositories SmartDukaan

Rev

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