feature: Claim check pattern azure (#1897)
* Archietecture Diagram added * Added pom.xml * Architecture Diagram Updated * Added Microservices and kafka * ReadME File Added * ReadME file Updated * #1329 ReadMe file updated and working pattern code added * #1329 readme file updated * #1329 readme file updated and java documentation added * #1329 repository merged * Update claim-check-pattern/ReadME.md #1329 Real world description updated Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Update claim-check-pattern/ReadME.md Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Applicability section in ReadMe file updated Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Tutorial section in ReadMe file updated Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Storage Data section in ReadMe File updated Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 workflow section Update claim-check-pattern/ReadME.md Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Update claim-check-pattern/pom.xml Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 deleted mvnw.cmd file * #1329 deleted drawio file * #1329 deleted mvnw file * #1329 deleted mvnw file * #1329 mvnw.cmd file deleted * #1329 Update claim-check-pattern/usage-cost-processor/src/main/java/com/callusage/domain/MessageHeader.java Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Update claim-check-pattern/usage-cost-processor/src/main/java/com/callusage/domain/UsageCostDetail.java Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Update claim-check-pattern/usage-cost-processor/src/main/java/com/callusage/domain/UsageDetail.java Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 deleted mvnw file * #1329 deleted mvnw.cmd file * #1329 Update claim-check-pattern/usage-detail-sender/src/main/java/com/callusage/domain/Message.java Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 Update claim-check-pattern/ReadME.md Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> * #1329 pom file dependencies fixed, readmeflie updated, removed unused imports * #1329 Readfile updated, class javadoc added * #1329 UML class diagrams added, readme file updated * #1329 lombok annotations used on model classes, common dependencies moved to parent pom * #1329 test cases added * include claim-check-pattern in parent pom * #1329 code smells fixed * #1329 code smells removed * #1329 security issues fixed * #1329 updated pom files and refactored packages name * #1329 checkstyle warning fixed * #1329 code coverage increased * #1329 minor changed. * checkpoint created with common utility * Claim-Check-Pattern | Shrirang97 | Implemented using Java Azure Functions * Claim-Check-Pattern | Shrirang97 | Updated Functions logic * Update MessageHandlerUtility.java * Update UsageCostProcessorFunction.java * claim-check-pattern | Shrirang97 | Added test cases * claim-check-pattern | Shrirang97 | Test cases for Azure functions fixed * Claim-Check-Pattern | Shrirang | Used string as request body * claim-check-pattern | Shrirang | Working test cases * claim-check-pattern | Shrirang | Issue fixed while deserializing * claim-check-pattern | Shrirang | removed unused import * claim-check-pattern | Shrirang | fixed refactoring * claim-chek-pattern | Shrirang | added lombok | fixed dependencies & test cases * Delete .DS_Store * claim-check-pattern | Shrirang | Fixed unrelated file * Update BookService.java * Update BookService.java * claim-check-pattern | Shrirang | Fixed unrelated files * claim-check-pattern | Shrirang | Fixed review comments * Update UsageCostProcessorFunction.java * Update ReadME.md * Rename ReadME.md to README.md * claim-check-pattern | Shrirang | Incorporated review comments. * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Update cloud-claim-check-pattern/README.md Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com> * Updated readme file * Read me file updated | Added more description Co-authored-by: Subhrodip Mohanta <subhrodipmohanta@gmail.com> Co-authored-by: Subhrodip Mohanta <hello@subho.xyz> Co-authored-by: Ilkka Seppälä <iluwatar@users.noreply.github.com>
This commit is contained in:
@@ -0,0 +1,168 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.consumer.callcostprocessor.functions;
|
||||
|
||||
import com.azure.core.util.BinaryData;
|
||||
import com.azure.core.util.serializer.TypeReference;
|
||||
import com.azure.messaging.eventgrid.EventGridEvent;
|
||||
import com.azure.messaging.eventgrid.systemevents.SubscriptionValidationEventData;
|
||||
import com.azure.messaging.eventgrid.systemevents.SubscriptionValidationResponse;
|
||||
import com.iluwatar.claimcheckpattern.domain.Message;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageBody;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageHeader;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageReference;
|
||||
import com.iluwatar.claimcheckpattern.domain.UsageCostDetail;
|
||||
import com.iluwatar.claimcheckpattern.domain.UsageDetail;
|
||||
import com.iluwatar.claimcheckpattern.utility.MessageHandlerUtility;
|
||||
import com.microsoft.azure.functions.ExecutionContext;
|
||||
import com.microsoft.azure.functions.HttpMethod;
|
||||
import com.microsoft.azure.functions.HttpRequestMessage;
|
||||
import com.microsoft.azure.functions.HttpResponseMessage;
|
||||
import com.microsoft.azure.functions.HttpStatus;
|
||||
import com.microsoft.azure.functions.annotation.AuthorizationLevel;
|
||||
import com.microsoft.azure.functions.annotation.FunctionName;
|
||||
import com.microsoft.azure.functions.annotation.HttpTrigger;
|
||||
import java.time.OffsetDateTime;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.Optional;
|
||||
|
||||
/**
|
||||
* Azure Functions with HTTP Trigger.
|
||||
* This is Consumer class.
|
||||
*/
|
||||
public class UsageCostProcessorFunction {
|
||||
|
||||
private MessageHandlerUtility<UsageDetail> messageHandlerUtilityForUsageDetail;
|
||||
private MessageHandlerUtility<UsageCostDetail> messageHandlerUtilityForUsageCostDetail;
|
||||
|
||||
public UsageCostProcessorFunction() {
|
||||
this.messageHandlerUtilityForUsageDetail = new MessageHandlerUtility<>();
|
||||
this.messageHandlerUtilityForUsageCostDetail = new MessageHandlerUtility<>();
|
||||
}
|
||||
|
||||
public UsageCostProcessorFunction(
|
||||
MessageHandlerUtility<UsageDetail> messageHandlerUtilityForUsageDetail,
|
||||
MessageHandlerUtility<UsageCostDetail> messageHandlerUtilityForUsageCostDetail) {
|
||||
this.messageHandlerUtilityForUsageDetail = messageHandlerUtilityForUsageDetail;
|
||||
this.messageHandlerUtilityForUsageCostDetail = messageHandlerUtilityForUsageCostDetail;
|
||||
}
|
||||
|
||||
/**
|
||||
* Azure function which gets triggered when event grid event send event to it.
|
||||
* After receiving event, it read input file from blob storage, calculate call cost details.
|
||||
* It creates new message with cost details and drop message to blob storage.
|
||||
* @param request represents HttpRequestMessage
|
||||
* @param context represents ExecutionContext
|
||||
* @return HttpResponseMessage
|
||||
*/
|
||||
@FunctionName("UsageCostProcessorFunction")
|
||||
public HttpResponseMessage run(@HttpTrigger(name = "req", methods = { HttpMethod.GET,
|
||||
HttpMethod.POST }, authLevel = AuthorizationLevel.ANONYMOUS)
|
||||
HttpRequestMessage<Optional<String>> request,
|
||||
final ExecutionContext context) {
|
||||
try {
|
||||
var eventGridEvents = EventGridEvent.fromString(request.getBody().get());
|
||||
for (var eventGridEvent : eventGridEvents) {
|
||||
// Handle system events
|
||||
if (eventGridEvent.getEventType()
|
||||
.equals("Microsoft.EventGrid.SubscriptionValidationEvent")) {
|
||||
SubscriptionValidationEventData subscriptionValidationEventData = eventGridEvent.getData()
|
||||
.toObject(SubscriptionValidationEventData.class);
|
||||
// Handle the subscription validation event
|
||||
var responseData = new SubscriptionValidationResponse();
|
||||
responseData.setValidationResponse(subscriptionValidationEventData.getValidationCode());
|
||||
return request.createResponseBuilder(HttpStatus.OK).body(responseData).build();
|
||||
|
||||
} else if (eventGridEvent.getEventType().equals("UsageDetail")) {
|
||||
// Get message header and reference
|
||||
var messageReference = eventGridEvent.getData()
|
||||
.toObject(MessageReference.class);
|
||||
|
||||
// Read message from persistent storage
|
||||
var message = this.messageHandlerUtilityForUsageDetail
|
||||
.readFromPersistantStorage(messageReference, context.getLogger());
|
||||
|
||||
// Get Data and generate cost details
|
||||
List<UsageDetail> usageDetailsList = BinaryData.fromObject(
|
||||
message.getMessageBody().getData())
|
||||
.toObject(new TypeReference<>() {
|
||||
});
|
||||
var usageCostDetailsList = calculateUsageCostDetails(usageDetailsList);
|
||||
|
||||
// Create message body
|
||||
var newMessageBody = new MessageBody<UsageCostDetail>();
|
||||
newMessageBody.setData(usageCostDetailsList);
|
||||
|
||||
// Create message header
|
||||
var newMessageReference = new MessageReference("callusageapp",
|
||||
eventGridEvent.getId() + "/output.json");
|
||||
var newMessageHeader = new MessageHeader();
|
||||
newMessageHeader.setId(eventGridEvent.getId());
|
||||
newMessageHeader.setSubject("UsageCostProcessor");
|
||||
newMessageHeader.setTopic("");
|
||||
newMessageHeader.setEventType("UsageCostDetail");
|
||||
newMessageHeader.setEventTime(OffsetDateTime.now().toString());
|
||||
newMessageHeader.setData(newMessageReference);
|
||||
newMessageHeader.setDataVersion("v1.0");
|
||||
|
||||
// Create entire message
|
||||
var newMessage = new Message<UsageCostDetail>();
|
||||
newMessage.setMessageHeader(newMessageHeader);
|
||||
newMessage.setMessageBody(newMessageBody);
|
||||
|
||||
// Drop data to persistent storage
|
||||
this.messageHandlerUtilityForUsageCostDetail.dropToPersistantStorage(newMessage,
|
||||
context.getLogger());
|
||||
|
||||
context.getLogger().info("Message is dropped successfully");
|
||||
return request.createResponseBuilder(HttpStatus.OK)
|
||||
.body("Message is dropped successfully").build();
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
context.getLogger().warning(e.getMessage());
|
||||
}
|
||||
|
||||
return request.createResponseBuilder(HttpStatus.INTERNAL_SERVER_ERROR).body(null).build();
|
||||
}
|
||||
|
||||
private List<UsageCostDetail> calculateUsageCostDetails(List<UsageDetail> usageDetailsList) {
|
||||
if (usageDetailsList == null) {
|
||||
return null;
|
||||
}
|
||||
var usageCostDetailsList = new ArrayList<UsageCostDetail>();
|
||||
|
||||
usageDetailsList.forEach(usageDetail -> {
|
||||
var usageCostDetail = new UsageCostDetail();
|
||||
usageCostDetail.setUserId(usageDetail.getUserId());
|
||||
usageCostDetail.setCallCost(usageDetail.getDuration() * 0.30); // 0.30₹ per minute
|
||||
usageCostDetail.setDataCost(usageDetail.getData() * 0.20); // 0.20₹ per MB
|
||||
|
||||
usageCostDetailsList.add(usageCostDetail);
|
||||
});
|
||||
|
||||
return usageCostDetailsList;
|
||||
}
|
||||
}
|
@@ -0,0 +1,43 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.domain;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* It is the message which gets dropped or read by Producer or Consumer Azure functions.
|
||||
* It is stored in the json format.
|
||||
* @param <T> represents UsageDetail or UsageCostDetail
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public class Message<T> {
|
||||
private MessageHeader messageHeader;
|
||||
|
||||
private MessageBody<T> messageBody;
|
||||
|
||||
}
|
@@ -0,0 +1,43 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.domain;
|
||||
|
||||
import java.util.List;
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* It is message body of the message.
|
||||
* It stores actual data in our case UsageCostDetail or UsageDetail.
|
||||
* @param <T> represents UsageDetail or UsageCostDetail
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public class MessageBody<T> {
|
||||
|
||||
private List<T> data;
|
||||
|
||||
}
|
@@ -0,0 +1,47 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.domain;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* This is message header or event which is sent to Event Grid.
|
||||
* Its structure is same as Azure Event Grid Event Class.
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public class MessageHeader {
|
||||
|
||||
private String id;
|
||||
private String subject;
|
||||
private String topic;
|
||||
private String eventType;
|
||||
private String eventTime;
|
||||
private Object data;
|
||||
private String dataVersion;
|
||||
|
||||
}
|
@@ -0,0 +1,45 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.domain;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* This is claim/message reference class.
|
||||
* It contains the information about data where it is stored in persistent storage
|
||||
* and file name.
|
||||
* dataLocation is blob storage container name.
|
||||
* dataFileName is file name in above container.
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public class MessageReference {
|
||||
|
||||
private String dataLocation;
|
||||
private String dataFileName;
|
||||
|
||||
}
|
@@ -0,0 +1,43 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.domain;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* This is call cost details class.
|
||||
* It stores userId of the caller, call duration cost and data cost.
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public class UsageCostDetail {
|
||||
|
||||
private String userId;
|
||||
private double callCost;
|
||||
private double dataCost;
|
||||
|
||||
}
|
@@ -0,0 +1,44 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.domain;
|
||||
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Data;
|
||||
import lombok.NoArgsConstructor;
|
||||
|
||||
/**
|
||||
* This is call usage detail calls.
|
||||
* It stores userId of the caller, call duration and data used.
|
||||
*/
|
||||
@Data
|
||||
@NoArgsConstructor
|
||||
@AllArgsConstructor
|
||||
public class UsageDetail {
|
||||
|
||||
private String userId;
|
||||
|
||||
private int duration;
|
||||
|
||||
private int data;
|
||||
}
|
@@ -0,0 +1,146 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.producer.calldetails.functions;
|
||||
|
||||
import com.azure.messaging.eventgrid.EventGridEvent;
|
||||
import com.azure.messaging.eventgrid.systemevents.SubscriptionValidationEventData;
|
||||
import com.azure.messaging.eventgrid.systemevents.SubscriptionValidationResponse;
|
||||
import com.iluwatar.claimcheckpattern.domain.Message;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageBody;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageHeader;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageReference;
|
||||
import com.iluwatar.claimcheckpattern.domain.UsageDetail;
|
||||
import com.iluwatar.claimcheckpattern.utility.EventHandlerUtility;
|
||||
import com.iluwatar.claimcheckpattern.utility.MessageHandlerUtility;
|
||||
import com.microsoft.azure.functions.ExecutionContext;
|
||||
import com.microsoft.azure.functions.HttpMethod;
|
||||
import com.microsoft.azure.functions.HttpRequestMessage;
|
||||
import com.microsoft.azure.functions.HttpResponseMessage;
|
||||
import com.microsoft.azure.functions.HttpStatus;
|
||||
import com.microsoft.azure.functions.annotation.AuthorizationLevel;
|
||||
import com.microsoft.azure.functions.annotation.FunctionName;
|
||||
import com.microsoft.azure.functions.annotation.HttpTrigger;
|
||||
import java.time.OffsetDateTime;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.Optional;
|
||||
import java.util.Random;
|
||||
import java.util.UUID;
|
||||
|
||||
/**
|
||||
* Azure Functions with HTTP Trigger.
|
||||
* This is Producer class.
|
||||
*/
|
||||
public class UsageDetailPublisherFunction {
|
||||
|
||||
private MessageHandlerUtility<UsageDetail> messageHandlerUtility;
|
||||
private EventHandlerUtility<MessageHeader> eventHandlerUtility;
|
||||
|
||||
public UsageDetailPublisherFunction() {
|
||||
this.messageHandlerUtility = new MessageHandlerUtility<>();
|
||||
this.eventHandlerUtility = new EventHandlerUtility<>();
|
||||
}
|
||||
|
||||
public UsageDetailPublisherFunction(MessageHandlerUtility<UsageDetail> messageHandlerUtility,
|
||||
EventHandlerUtility<MessageHeader> eventHandlerUtility) {
|
||||
this.messageHandlerUtility = messageHandlerUtility;
|
||||
this.eventHandlerUtility = eventHandlerUtility;
|
||||
}
|
||||
|
||||
/**
|
||||
* Azure function which create message, drop it in persistent storage
|
||||
* and publish the event to Event Grid topic.
|
||||
* @param request represents HttpRequestMessage
|
||||
* @param context represents ExecutionContext
|
||||
* @return HttpResponseMessage
|
||||
*/
|
||||
@FunctionName("UsageDetailPublisherFunction")
|
||||
public HttpResponseMessage run(@HttpTrigger(name = "req", methods = {
|
||||
HttpMethod.POST }, authLevel = AuthorizationLevel.ANONYMOUS)
|
||||
HttpRequestMessage<Optional<String>> request,
|
||||
final ExecutionContext context) {
|
||||
try {
|
||||
|
||||
var eventGridEvents = EventGridEvent.fromString(request.getBody().get());
|
||||
|
||||
for (EventGridEvent eventGridEvent : eventGridEvents) {
|
||||
// Handle system events
|
||||
if (eventGridEvent.getEventType()
|
||||
.equals("Microsoft.EventGrid.SubscriptionValidationEvent")) {
|
||||
SubscriptionValidationEventData subscriptionValidationEventData = eventGridEvent.getData()
|
||||
.toObject(SubscriptionValidationEventData.class);
|
||||
// Handle the subscription validation event
|
||||
var responseData = new SubscriptionValidationResponse();
|
||||
responseData.setValidationResponse(subscriptionValidationEventData.getValidationCode());
|
||||
return request.createResponseBuilder(HttpStatus.OK).body(responseData).build();
|
||||
|
||||
} else if (eventGridEvent.getEventType().equals("UsageDetail")) {
|
||||
// Create message body
|
||||
var messageBody = new MessageBody<UsageDetail>();
|
||||
var usageDetailsList = new ArrayList<UsageDetail>();
|
||||
var random = new Random();
|
||||
for (int i = 0; i < 51; i++) {
|
||||
var usageDetail = new UsageDetail();
|
||||
usageDetail.setUserId("userId" + i);
|
||||
usageDetail.setData(random.nextInt(500));
|
||||
usageDetail.setDuration(random.nextInt(500));
|
||||
|
||||
usageDetailsList.add(usageDetail);
|
||||
}
|
||||
messageBody.setData(usageDetailsList);
|
||||
|
||||
// Create message header
|
||||
var messageHeader = new MessageHeader();
|
||||
messageHeader.setId(UUID.randomUUID().toString());
|
||||
messageHeader.setSubject("UsageDetailPublisher");
|
||||
messageHeader.setTopic("usagecostprocessorfunction-topic");
|
||||
messageHeader.setEventType("UsageDetail");
|
||||
messageHeader.setEventTime(OffsetDateTime.now().toString());
|
||||
var messageReference = new MessageReference("callusageapp",
|
||||
messageHeader.getId() + "/input.json");
|
||||
messageHeader.setData(messageReference);
|
||||
messageHeader.setDataVersion("v1.0");
|
||||
|
||||
// Create entire message
|
||||
var message = new Message<UsageDetail>();
|
||||
message.setMessageHeader(messageHeader);
|
||||
message.setMessageBody(messageBody);
|
||||
|
||||
// Drop data to persistent storage
|
||||
this.messageHandlerUtility.dropToPersistantStorage(message, context.getLogger());
|
||||
|
||||
// Publish event to event grid topic
|
||||
eventHandlerUtility.publishEvent(messageHeader, context.getLogger());
|
||||
|
||||
context.getLogger().info("Message is dropped and event is published successfully");
|
||||
return request.createResponseBuilder(HttpStatus.OK).body(message).build();
|
||||
}
|
||||
}
|
||||
} catch (Exception e) {
|
||||
context.getLogger().warning(e.getMessage());
|
||||
}
|
||||
|
||||
return request.createResponseBuilder(HttpStatus.OK).body(null).build();
|
||||
}
|
||||
}
|
@@ -0,0 +1,66 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.utility;
|
||||
|
||||
import com.azure.core.credential.AzureKeyCredential;
|
||||
import com.azure.core.util.BinaryData;
|
||||
import com.azure.messaging.eventgrid.EventGridPublisherClient;
|
||||
import com.azure.messaging.eventgrid.EventGridPublisherClientBuilder;
|
||||
import java.util.logging.Logger;
|
||||
|
||||
/**
|
||||
* This class is event publisher utility which published message header to Event Grid topic.
|
||||
* @param <T> represents UsageDetail or UsageCostDetail
|
||||
*/
|
||||
public class EventHandlerUtility<T> {
|
||||
|
||||
private EventGridPublisherClient<BinaryData> customEventClient;
|
||||
|
||||
/** Default constructor.
|
||||
*/
|
||||
public EventHandlerUtility() {
|
||||
this.customEventClient = new EventGridPublisherClientBuilder()
|
||||
.endpoint(System.getenv("EventGridURL"))
|
||||
.credential(new AzureKeyCredential(System.getenv("EventGridKey")))
|
||||
.buildCustomEventPublisherClient();
|
||||
}
|
||||
|
||||
/**
|
||||
Parameterized constructor.
|
||||
*/
|
||||
public EventHandlerUtility(EventGridPublisherClient<BinaryData> customEventClient) {
|
||||
this.customEventClient = customEventClient;
|
||||
}
|
||||
|
||||
/**
|
||||
Method for publishing event to Event Grid Topic.
|
||||
*/
|
||||
public void publishEvent(T customEvent, Logger logger) {
|
||||
try {
|
||||
customEventClient.sendEvent(BinaryData.fromObject(customEvent));
|
||||
} catch (Exception e) {
|
||||
logger.info(e.getMessage());
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,127 @@
|
||||
/*
|
||||
* The MIT License
|
||||
* Copyright © 2014-2021 Ilkka Seppälä
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
* of this software and associated documentation files (the "Software"), to deal
|
||||
* in the Software without restriction, including without limitation the rights
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
* copies of the Software, and to permit persons to whom the Software is
|
||||
* furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
* THE SOFTWARE.
|
||||
*/
|
||||
|
||||
package com.iluwatar.claimcheckpattern.utility;
|
||||
|
||||
import com.azure.core.util.BinaryData;
|
||||
import com.azure.core.util.serializer.TypeReference;
|
||||
import com.azure.storage.blob.BlobClient;
|
||||
import com.azure.storage.blob.BlobContainerClient;
|
||||
import com.azure.storage.blob.BlobServiceClient;
|
||||
import com.azure.storage.blob.BlobServiceClientBuilder;
|
||||
import com.iluwatar.claimcheckpattern.domain.Message;
|
||||
import com.iluwatar.claimcheckpattern.domain.MessageReference;
|
||||
import java.util.logging.Logger;
|
||||
|
||||
/**
|
||||
* This class read and drop message from Azure blob storage.
|
||||
* @param <T> represents UsageDetail or UsageCostDetail
|
||||
*/
|
||||
public class MessageHandlerUtility<T> {
|
||||
|
||||
private BlobServiceClient blobServiceClient;
|
||||
|
||||
/**
|
||||
* Parameterized constructor.
|
||||
* @param blobServiceClient represents BlobServiceClient
|
||||
*/
|
||||
public MessageHandlerUtility(BlobServiceClient blobServiceClient) {
|
||||
this.blobServiceClient = blobServiceClient;
|
||||
}
|
||||
|
||||
/**
|
||||
* Default constructor.
|
||||
*/
|
||||
public MessageHandlerUtility() {
|
||||
// Create a BlobServiceClient object which will be used to create a container
|
||||
// client
|
||||
this.blobServiceClient = new BlobServiceClientBuilder()
|
||||
.connectionString(System.getenv("BlobStorageConnectionString")).buildClient();
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Read message from blob storage.
|
||||
* @param messageReference represents MessageReference
|
||||
* @param logger represents Logger
|
||||
* @return Message
|
||||
*/
|
||||
public Message<T> readFromPersistantStorage(MessageReference messageReference, Logger logger) {
|
||||
Message<T> message = null;
|
||||
try {
|
||||
|
||||
// Get container name from message reference
|
||||
String containerName = messageReference.getDataLocation();
|
||||
|
||||
// Get blob name from message reference
|
||||
String blobName = messageReference.getDataFileName();
|
||||
|
||||
// Get container client
|
||||
BlobContainerClient containerClient = blobServiceClient.getBlobContainerClient(containerName);
|
||||
|
||||
// Get a reference to a blob
|
||||
BlobClient blobClient = containerClient.getBlobClient(blobName);
|
||||
|
||||
// download the blob
|
||||
message = blobClient.downloadContent().toObject(new TypeReference<Message<T>>() {
|
||||
});
|
||||
} catch (Exception e) {
|
||||
logger.info(e.getMessage());
|
||||
}
|
||||
return message;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Drop message to blob storage.
|
||||
* @param message represents Message
|
||||
* @param logger represents Logger
|
||||
*/
|
||||
public void dropToPersistantStorage(Message<T> message, Logger logger) {
|
||||
try {
|
||||
|
||||
// Get message reference
|
||||
MessageReference messageReference = (MessageReference) message.getMessageHeader().getData();
|
||||
|
||||
// Create a unique name for the container
|
||||
String containerName = messageReference.getDataLocation();
|
||||
|
||||
// Create the container and return a container client object
|
||||
BlobContainerClient containerClient = this.blobServiceClient
|
||||
.getBlobContainerClient(containerName);
|
||||
if (!containerClient.exists()) {
|
||||
containerClient.create();
|
||||
}
|
||||
|
||||
// Get a reference to a blob
|
||||
BlobClient blobClient = containerClient.getBlobClient(messageReference.getDataFileName());
|
||||
|
||||
// Upload the blob
|
||||
blobClient.upload(BinaryData.fromObject(message));
|
||||
} catch (Exception e) {
|
||||
logger.info(e.getMessage());
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}
|
Reference in New Issue
Block a user