diff --git a/notification-core/src/main/java/org/opengroup/osdu/notification/api/PubsubEndpoint.java b/notification-core/src/main/java/org/opengroup/osdu/notification/api/PubsubEndpoint.java
index 935ecaa241bc9774f720570af83de6ddcca15fa4..510af470777c68dd4707df91df099b445b58ca1a 100644
--- a/notification-core/src/main/java/org/opengroup/osdu/notification/api/PubsubEndpoint.java
+++ b/notification-core/src/main/java/org/opengroup/osdu/notification/api/PubsubEndpoint.java
@@ -87,7 +87,7 @@ public class PubsubEndpoint {
     @PreAuthorize("@authorizationFilter.hasAnyPermission('" + Config.OPS + "', '" + Config.PUBSUB + "')")
     public ResponseEntity recordChanged() throws Exception {
         if(this.pubsubRequestBodyExtractor.isHandshakeRequest()) {
-            this.pubsubHandshakeHandler.getHandshakeResponse();
+            return ResponseEntity.ok(this.pubsubHandshakeHandler.getHandshakeResponse());
         }
 
         String notificationId = this.pubsubRequestBodyExtractor.extractNotificationIdFromRequestBody();
diff --git a/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridHandshakeHandler.java b/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridHandshakeHandler.java
index f9cff0fefd0578bba8a303f18ef5b8ef7b2e8075..065ffca5ef72cfd9dd0028b1c74c0c7885ac9c94 100644
--- a/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridHandshakeHandler.java
+++ b/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridHandshakeHandler.java
@@ -14,52 +14,39 @@
 
 package org.opengroup.osdu.notification.provider.azure.pubsub;
 
-import com.google.gson.JsonArray;
-import com.google.gson.JsonElement;
 import com.google.gson.JsonObject;
-import com.google.gson.JsonParser;
 import org.opengroup.osdu.core.common.model.http.AppException;
+import org.opengroup.osdu.notification.provider.azure.util.RequestAttributesDispatcher;
 import org.opengroup.osdu.notification.pubsub.IPubsubHandshakeHandler;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.context.annotation.Lazy;
 import org.springframework.http.HttpStatus;
-import org.springframework.http.ResponseEntity;
 import org.springframework.stereotype.Component;
 
-import javax.servlet.http.HttpServletRequest;
-import java.io.BufferedReader;
-import java.io.IOException;
-import java.util.stream.Collectors;
-import java.util.stream.Stream;
-
 @Component
 @Lazy
 public class EventGridHandshakeHandler implements IPubsubHandshakeHandler {
 
-    private JsonObject root = null;
-
     @Autowired
-    private HttpServletRequest request;
+    private RequestAttributesDispatcher requestDispatcher;
 
     /**
      * Extract Handshake response string form  Handshake request.
-     *
+     * TODO: Check if there is a need to verify subscription name with
+     *  registration service, before verifying the endpoint.
      * @return validation string
      */
     @Override
     public String getHandshakeResponse() {
-        if (this.root == null) {
-            this.root = this.extractRootJsonElementFromRequestBody();
-        }
-
-        String response = null;
+         String response = null;
 
         try {
-            JsonObject data = (JsonObject) this.root.get("data");
+            JsonObject requestRoot = requestDispatcher.getRequestRoot();
+            JsonObject data = (JsonObject) requestRoot.get("data");
             String validationCode = data.get("validationCode").getAsString();
             JsonObject jsonResponse = new JsonObject();
             jsonResponse.addProperty("ValidationResponse", validationCode);
-            response = validationCode.toString();
+            response = jsonResponse.toString();
         } catch (Exception exception) {
             throw new AppException(HttpStatus.INTERNAL_SERVER_ERROR.value(), "Request payload parsing error",
                     "Unable to parse request payload.", exception);
@@ -67,29 +54,4 @@ public class EventGridHandshakeHandler implements IPubsubHandshakeHandler {
 
         return response;
     }
-
-    /**
-     * Utility method to extract root element from the request.
-     *
-     * @return Root JsonObject
-     */
-    private JsonObject extractRootJsonElementFromRequestBody() {
-        try {
-            JsonParser jsonParser = new JsonParser();
-            BufferedReader reader = request.getReader();
-            Stream<String> lines = reader.lines();
-            String requestBody = lines.collect(Collectors.joining("\n"));
-            JsonArray requestArray = (JsonArray) jsonParser.parse(requestBody);
-            JsonElement rootElement = requestArray.get(0);
-            if (!(rootElement instanceof JsonObject)) {
-                throw new AppException(HttpStatus.BAD_REQUEST.value(), "RequestBody is not JsonObject.",
-                        "Request Body should be JsonObject to be processed.");
-            }
-
-            return rootElement.getAsJsonObject();
-        } catch (IOException e) {
-            throw new AppException(HttpStatus.INTERNAL_SERVER_ERROR.value(), "Request payload parsing error",
-                    "Unable to parse request payload.", e);
-        }
-    }
 }
diff --git a/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridRequestBodyExtractor.java b/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridRequestBodyExtractor.java
index 73a6c3219fbb291a87d9f3a6d4aee1d663a002ac..bd4464efc9e3e79de17833fade4820359f48af10 100644
--- a/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridRequestBodyExtractor.java
+++ b/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/pubsub/EventGridRequestBodyExtractor.java
@@ -18,40 +18,44 @@ import com.google.common.base.Strings;
 import com.google.gson.Gson;
 import com.google.gson.JsonElement;
 import com.google.gson.JsonObject;
-import com.google.gson.JsonParser;
 import org.opengroup.osdu.core.common.logging.JaxRsDpsLog;
 import org.opengroup.osdu.core.common.model.http.AppException;
 import org.opengroup.osdu.notification.models.MessageContent;
+import org.opengroup.osdu.notification.provider.azure.util.RequestAttributesDispatcher;
 import org.opengroup.osdu.notification.pubsub.IPubsubRequestBodyExtractor;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.http.HttpStatus;
 import org.springframework.stereotype.Component;
 import org.springframework.web.context.annotation.RequestScope;
 
-import javax.servlet.http.HttpServletRequest;
-import java.io.BufferedReader;
-import java.io.IOException;
 import java.util.Base64;
 import java.util.HashMap;
 import java.util.Map;
-import java.util.stream.Collectors;
-import java.util.stream.Stream;
+
+// TODO : Make error messages as constants.
+// TODO : Add TracePoints to exception logs so that monitoring is done well.
 
 @Component
 @RequestScope
 public class EventGridRequestBodyExtractor implements IPubsubRequestBodyExtractor {
-    private static final String INVALID_PUBSUB_MESSAGE = "Invalid pubsub message";
+    private static final String INVALID_EVENTGRID_MESSAGE = "Invalid Event Grid Message";
     private static final String EVENTGRID_VALIDATION_EVENT = "Microsoft.EventGrid.SubscriptionValidationEvent";
     private static final Gson GSON = new Gson();
     private MessageContent messageContent;
     private JsonObject root = null;
 
     @Autowired
-    private HttpServletRequest request;
+    private RequestAttributesDispatcher requestDispatcher;
 
     @Autowired
     private JaxRsDpsLog log;
 
+    /**
+     * Extracts the attributes from the request that are filled in by publisher of the message.
+     *
+     * @throws AppException
+     * @return Request Attributes Map
+     */
     public Map<String, String> extractAttributesFromRequestBody() {
         if (this.messageContent == null) {
             this.messageContent = this.extractPubsubMessageFromRequestBody();
@@ -59,6 +63,12 @@ public class EventGridRequestBodyExtractor implements IPubsubRequestBodyExtracto
         return this.messageContent.getAttributes();
     }
 
+    /**
+     * Extracts the data from the request that are filled in by publisher of the message,
+     *
+     * @throws AppException
+     * @return Request Data String
+     */
     public String extractDataFromRequestBody() {
         if (this.messageContent == null) {
             this.messageContent = this.extractPubsubMessageFromRequestBody();
@@ -66,49 +76,70 @@ public class EventGridRequestBodyExtractor implements IPubsubRequestBodyExtracto
         return this.messageContent.getData();
     }
 
+    /**
+     * Extracts the notificationId from the request that are filled in by EventGrid.
+     *
+     * @throws AppException
+     * @return Request NotificationId String.
+     */
     public String extractNotificationIdFromRequestBody() {
+        return this.requestDispatcher.getRequestSubscriptionId();
     }
 
+    /**
+     * Checks if the request is for handshake.
+     *
+     * @throws AppException
+     * @return Request Type Boolean
+     */
     public boolean isHandshakeRequest() {
         if (this.root == null) {
-            this.root = this.extractRootJsonElementFromRequestBody();
+            this.root = this.requestDispatcher.getRequestRoot();
         }
-        try {
-            JsonElement data = this.root.get("eventType");
-            if (EVENTGRID_VALIDATION_EVENT.equals(data.getAsString())) {
-                return true;
-            }
-        } catch (Exception exception) {
-            throw new AppException(HttpStatus.BAD_REQUEST.value(), "RequestBody is not JsonObject.",
-                    "Request Body should be JsonObject to be processed.");
+        JsonElement data = this.root.get("eventType");
+        if (EVENTGRID_VALIDATION_EVENT.equals(data.getAsString())) {
+             return true;
         }
+
         return false;
     }
 
+    /**
+     * Utility method that wxtracts the core content in the request.
+     * This is what the publisher will publish.
+     *
+     * @throws AppException
+     * @return Message Content Object
+     */
     private MessageContent extractPubsubMessageFromRequestBody() {
-
-       /* if (this.root == null) {
-            this.root = this.extractRootJsonElementFromRequestBody();
+        if (this.root == null) {
+            this.root = this.requestDispatcher.getRequestRoot();
         }
-        JsonElement message = this.root.get("message");
+        JsonElement message = this.root.get("data");
         if (message == null) {
-            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_PUBSUB_MESSAGE, "message object not found");
+            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_EVENTGRID_MESSAGE, "message object not found");
         }
+
         MessageContent content = GSON.fromJson(message.toString(), MessageContent.class);
+        String eventTime = this.root.getAsJsonObject().get("eventTime").getAsString();
+        if (Strings.isNullOrEmpty(eventTime)) {
+            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_EVENTGRID_MESSAGE, "Event time not found");
+        }
+        content.setPublishTime(eventTime);
 
         Map<String, String> attributes = content.getAttributes();
         if (attributes == null || attributes.isEmpty()) {
             log.error("Incorrect Message: " + message.toString() );
-            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_PUBSUB_MESSAGE, "attribute map not found");
+            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_EVENTGRID_MESSAGE, "attribute map not found");
         }
         String data = content.getData();
         if (Strings.isNullOrEmpty(data)) {
-            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_PUBSUB_MESSAGE, "data field not found");
+            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_EVENTGRID_MESSAGE, "data field not found");
         }
         Map<String, String> lowerCase = new HashMap<>();
         attributes.forEach((key, value) -> lowerCase.put(key.toLowerCase(), value));
         if (Strings.isNullOrEmpty(attributes.get("data-partition-id"))) {
-            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_PUBSUB_MESSAGE,
+            throw new AppException(HttpStatus.BAD_REQUEST.value(), INVALID_EVENTGRID_MESSAGE,
                     "No tenant information from pubsub message.");
         }
         content.setAttributes(lowerCase);
@@ -116,24 +147,8 @@ public class EventGridRequestBodyExtractor implements IPubsubRequestBodyExtracto
         String decoded = new String(Base64.getDecoder().decode(data));
         content.setData(decoded);
 
-        return content;*/
+        return content;
     }
 
-    private JsonObject extractRootJsonElementFromRequestBody() {
-        try {
-            JsonParser jsonParser = new JsonParser();
-            BufferedReader reader = request.getReader();
-            Stream<String> lines = reader.lines();
-            String requestBody = lines.collect(Collectors.joining("\n"));
-            JsonElement rootElement = jsonParser.parse(requestBody);
-            if (!(rootElement instanceof JsonObject)) {
-                throw new AppException(HttpStatus.BAD_REQUEST.value(), "RequestBody is not JsonObject.",
-                        "Request Body should be JsonObject to be processed.");
-            }
-            return rootElement.getAsJsonObject();
-        } catch (IOException e) {
-            throw new AppException(HttpStatus.INTERNAL_SERVER_ERROR.value(), "Request payload parsing error",
-                    "Unable to parse request payload.", e);
-        }
-    }
+
 }
diff --git a/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/util/RequestAttributesDispatcher.java b/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/util/RequestAttributesDispatcher.java
new file mode 100644
index 0000000000000000000000000000000000000000..68f0e1d93b7705a07c3ae293c10380a4a1a0f677
--- /dev/null
+++ b/provider/notification-azure/src/main/java/org/opengroup/osdu/notification/provider/azure/util/RequestAttributesDispatcher.java
@@ -0,0 +1,106 @@
+// Copyright © Microsoft Corporation
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//      http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package org.opengroup.osdu.notification.provider.azure.util;
+
+import com.google.common.base.Strings;
+import com.google.gson.JsonArray;
+import com.google.gson.JsonElement;
+import com.google.gson.JsonObject;
+import com.google.gson.JsonParser;
+import org.opengroup.osdu.core.common.model.http.AppException;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.http.HttpStatus;
+import org.springframework.stereotype.Component;
+import org.springframework.web.context.annotation.RequestScope;
+
+import javax.servlet.http.HttpServletRequest;
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
+
+@Component
+@RequestScope
+public class RequestAttributesDispatcher {
+
+    private JsonObject requestRoot;
+
+    @Autowired
+    private HttpServletRequest request;
+
+    /**
+     * Extracts Request's root.
+     *
+     * @throws AppException
+     * @throws IOException
+     * @return Request Root JsonObject
+     */
+    public JsonObject getRequestRoot(){
+        if(requestRoot == null) {
+            try {
+                JsonArray requestArray = getRequest();
+                JsonElement rootElement = requestArray.get(0);
+                if (!(rootElement instanceof JsonObject)) {
+                    throw new AppException(HttpStatus.BAD_REQUEST.value(), "RequestBody is not JsonObject.",
+                            "Request Body should be JsonObject to be processed.");
+                }
+                this.requestRoot = rootElement.getAsJsonObject();
+
+            } catch (Exception e) {
+                throw new AppException(HttpStatus.INTERNAL_SERVER_ERROR.value(), "Request payload parsing error",
+                        "Unable to parse request payload.", e);
+            }
+        }
+
+        return requestRoot;
+    }
+
+    /**
+     * Extracts Subscription Name from the header.
+     *
+     * @throws AppException
+     * @throws IOException
+     * @return Sunscription Name String
+     */
+    public String getRequestSubscriptionId(){
+        String subscriptionId = request.getHeader("Aeg-Subscription-Name");
+        if (Strings.isNullOrEmpty(subscriptionId)) {
+            throw new AppException(HttpStatus.BAD_REQUEST.value(), "Invalid Event Grid Message", "Subscription ID not found");
+        }
+        return subscriptionId;
+    }
+
+    /**
+     * Utility method to read the request's body as stream and
+     * convert it into JsonArray
+     * @throws AppException
+     * @throws IOException
+     * @return Request JsonArray
+     */
+    private JsonArray getRequest() throws IOException {
+        JsonParser jsonParser = new JsonParser();
+        JsonArray requestArray;
+        try {
+            BufferedReader reader = request.getReader();
+            Stream<String> lines = reader.lines();
+            String requestBody = lines.collect(Collectors.joining("\n"));
+            requestArray = (JsonArray) jsonParser.parse(requestBody);
+        } catch (Exception e){
+            throw new AppException(HttpStatus.INTERNAL_SERVER_ERROR.value(), "Request payload parsing error",
+                    "Unable to read from the request.", e);
+        }
+        return requestArray;
+    }
+}
diff --git a/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/pubsub/EventGridHandshakeHandlerTest.java b/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/pubsub/EventGridHandshakeHandlerTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..abb2df02cbeee2b641a750e1025a242c4628fa18
--- /dev/null
+++ b/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/pubsub/EventGridHandshakeHandlerTest.java
@@ -0,0 +1,102 @@
+// Copyright © Microsoft Corporation
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//      http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package org.opengroup.osdu.notification.pubsub;
+
+import com.google.gson.JsonObject;
+import com.google.gson.JsonParser;
+import org.junit.Assert;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.mockito.InjectMocks;
+import org.mockito.Mock;
+import org.mockito.Spy;
+import org.mockito.junit.MockitoJUnitRunner;
+import org.opengroup.osdu.core.common.model.http.AppException;
+import org.opengroup.osdu.notification.provider.azure.pubsub.EventGridHandshakeHandler;
+import org.opengroup.osdu.notification.provider.azure.util.RequestAttributesDispatcher;
+import org.springframework.http.HttpStatus;
+
+import static org.junit.Assert.fail;
+import static org.mockito.Mockito.when;
+
+
+@RunWith(MockitoJUnitRunner.class)
+public class EventGridHandshakeHandlerTest {
+    @Mock
+    RequestAttributesDispatcher requestAttributesDispatcherMock;
+
+    @InjectMocks
+    @Spy
+    private EventGridHandshakeHandler sut;
+
+    @Test
+    public void should_returnValidResponse_getHandshakeResponse() {
+        // Set up
+        String validHandshakeRequestRoot =
+                "    {\n" +
+                        "        \"id\": \"testId\",\n" +
+                        "        \"topic\": \"testTopic\",\n" +
+                        "        \"subject\": \"\",\n" +
+                        "        \"data\": {\n" +
+                        "            \"validationCode\": \"testValidationCode\",\n" +
+                        "            \"validationUrl\": \"testURL\"\n" +
+                        "        },\n" +
+                        "        \"eventType\": \"Microsoft.EventGrid.SubscriptionValidationEvent\",\n" +
+                        "        \"eventTime\": \"2020-08-14T11:18:55.9278057Z\",\n" +
+                        "        \"metadataVersion\": \"1\",\n" +
+                        "        \"dataVersion\": \"2\"\n" +
+                        "    }";
+        JsonObject requestRoot = new JsonParser().parse(validHandshakeRequestRoot).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+        String expectedResponse = "{\"ValidationResponse\":\"testValidationCode\"}";
+
+        // Act
+        String observedResponse = this.sut.getHandshakeResponse();
+
+        // Assert
+        Assert.assertEquals(observedResponse, expectedResponse);
+    }
+
+    @Test
+    public void should_throwWhenDataIsMissing_getHandshakeResponse() {
+        // Set up
+        String validHandshakeRequestRoot =
+                "    {\n" +
+                        "        \"id\": \"testId\",\n" +
+                        "        \"topic\": \"testTopic\",\n" +
+                        "        \"subject\": \"\",\n" +
+                        "        \"eventType\": \"Microsoft.EventGrid.SubscriptionValidationEvent\",\n" +
+                        "        \"eventTime\": \"2020-08-14T11:18:55.9278057Z\",\n" +
+                        "        \"metadataVersion\": \"1\",\n" +
+                        "        \"dataVersion\": \"2\"\n" +
+                        "    }";
+        JsonObject requestRoot = new JsonParser().parse(validHandshakeRequestRoot).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        try {
+            // Act
+            this.sut.getHandshakeResponse();
+
+            // Asset
+            fail("Should Throw Exception");
+        } catch (
+        AppException appException){
+            Assert.assertEquals(HttpStatus.INTERNAL_SERVER_ERROR.value(), appException.getError().getCode());
+            Assert.assertEquals("Unable to parse request payload."  , appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+}
diff --git a/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/pubsub/EventGridRequestBodyExtractorTest.java b/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/pubsub/EventGridRequestBodyExtractorTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..c84e57c3b2201e3d67c20019fd11c3458431dbbf
--- /dev/null
+++ b/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/pubsub/EventGridRequestBodyExtractorTest.java
@@ -0,0 +1,294 @@
+// Copyright © Microsoft Corporation
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//      http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package org.opengroup.osdu.notification.pubsub;
+
+import com.google.gson.JsonObject;
+import com.google.gson.JsonParser;
+import org.junit.Assert;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.mockito.InjectMocks;
+import org.mockito.Mock;
+import org.mockito.Spy;
+import org.mockito.junit.MockitoJUnitRunner;
+import org.opengroup.osdu.core.common.model.http.AppException;
+import org.opengroup.osdu.notification.provider.azure.pubsub.EventGridRequestBodyExtractor;
+import org.opengroup.osdu.notification.provider.azure.util.RequestAttributesDispatcher;
+import org.springframework.http.HttpStatus;
+
+import java.util.Map;
+
+import static org.junit.Assert.fail;
+import static org.mockito.Mockito.when;
+
+@RunWith(MockitoJUnitRunner.class)
+public class EventGridRequestBodyExtractorTest {
+    private static final String INVALID_EVENTGRID_MESSAGE = "Invalid Event Grid Message";
+
+    @Mock
+    RequestAttributesDispatcher requestAttributesDispatcherMock;
+
+    @InjectMocks
+    @Spy
+    private EventGridRequestBodyExtractor sut;
+
+    @Test
+    public void should_returntrue_isHandshakeRequest() {
+        // Set up
+        String validHandshakeRequestRoot =
+                "    {\n" +
+                        "        \"id\": \"testId\",\n" +
+                        "        \"topic\": \"testTopic\",\n" +
+                        "        \"subject\": \"\",\n" +
+                        "        \"data\": {\n" +
+                        "            \"validationCode\": \"testValidationCode\",\n" +
+                        "            \"validationUrl\": \"testURL\"\n" +
+                        "        },\n" +
+                        "        \"eventType\": \"Microsoft.EventGrid.SubscriptionValidationEvent\",\n" +
+                        "        \"eventTime\": \"2020-08-14T11:18:55.9278057Z\",\n" +
+                        "        \"metadataVersion\": \"1\",\n" +
+                        "        \"dataVersion\": \"2\"\n" +
+                        "    }";
+        JsonObject requestRoot = new JsonParser().parse(validHandshakeRequestRoot).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        // Act
+        boolean response = this.sut.isHandshakeRequest();
+
+        // Assert
+        Assert.assertTrue(response);
+    }
+
+    @Test
+    public void should_returnfalse_isHandshakeRequest() {
+        //SetUp
+        String invalidHandshakeRequestRoot =
+                "    {\n" +
+                        "        \"id\": \"testId\",\n" +
+                        "        \"topic\": \"testTopic\",\n" +
+                        "        \"subject\": \"\",\n" +
+                        "        \"data\": {\n" +
+                        "            \"validationUrl\": \"testURL\"\n" +
+                        "        },\n" +
+                        "        \"eventType\": \"SubscriptionValidationEvent\",\n" +
+                        "        \"eventTime\": \"2020-08-14T11:18:55.9278057Z\",\n" +
+                        "        \"metadataVersion\": \"1\",\n" +
+                        "        \"dataVersion\": \"2\"\n" +
+                        "    }";
+        JsonObject requestRoot = new JsonParser().parse(invalidHandshakeRequestRoot).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        // Act
+        boolean response = this.sut.isHandshakeRequest();
+
+        // Assert
+        Assert.assertFalse(response);
+    }
+
+    @Test
+    public void should_throwWhenMessageIsMissing_extractDataFromRequestBody() {
+        // SetUp
+        String requestRootWithoutData = "{\n" +
+                "        \"id\": \"2425\",\n" +
+                "        \"eventType\": \"recordInserted\",\n" +
+                "        \"subject\": \"myapp/vehicles/motorcycles\",\n" +
+                "        \"dataVersion\": \"1.0\",\n" +
+                "        \"metadataVersion\": \"1\",\n" +
+                "        \"eventTime\": \"2020-08-14T18:04:12+00:00\",\n" +
+                "        \"topic\": \"testTopic\"\n" +
+                "    }";
+        JsonObject requestRoot = new JsonParser().parse(requestRootWithoutData).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        try{
+            // Act
+            this.sut.extractDataFromRequestBody();
+
+            // Asset
+            fail("Should Throw Exception");
+        } catch (AppException appException){
+            Assert.assertEquals(HttpStatus.BAD_REQUEST.value(), appException.getError().getCode());
+            Assert.assertEquals("message object not found", appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+
+    @Test
+    public void should_throwWhenAttributesAreMissing_extractDataFromRequestBody() {
+        String requestRootWithoutAttributes = "{\n" +
+                "        \"id\": \"2425\",\n" +
+                "        \"eventType\": \"recordInserted\",\n" +
+                "        \"subject\": \"myapp/vehicles/motorcycles\",\n" +
+                "        \"data\": {\n" +
+                "            \"data\": \"W3sia2luZCI6InRlc3RraW5kIiwiaWQiOiJ0ZXN0aWQiLCJvcGVyYXRpb250eXBlIjoiY3JlYXRlIn0seyJraW5kIjoidGVzdGtpbmQyIiwiaWQiOiJ0ZXN0aWQyIiwib3BlcmF0aW9udHlwZSI6InVwZGF0ZSJ9XQ\",\n" +
+                "            \"messageId\": \"136969346945\"\n" +
+                "        },\n" +
+                "        \"dataVersion\": \"1.0\",\n" +
+                "        \"metadataVersion\": \"1\",\n" +
+                "        \"eventTime\": \"2020-08-14T18:04:12+00:00\",\n" +
+                "        \"topic\": \"/subscriptions/c99e2bf3-1777-412b-baba-d823676589c2/resourceGroups/komakkar-OSDU-RG/providers/Microsoft.EventGrid/topics/recordChanged\"\n" +
+                "    }";
+        JsonObject requestRoot = new JsonParser().parse(requestRootWithoutAttributes).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        try{
+            // Act
+            this.sut.extractDataFromRequestBody();
+
+            // Asset
+            fail("Should Throw Exception");
+        } catch (AppException appException){
+            Assert.assertEquals(HttpStatus.BAD_REQUEST.value(), appException.getError().getCode());
+            Assert.assertEquals("attribute map not found", appException.getError().getMessage());
+        } catch (Exception exception) {
+            //fail("Should Throw AppException");
+        }
+    }
+
+    @Test
+    public void should_throwWhenDataFiledIsMissing_extractDataFromRequestBody() {
+        String requestRootWithoutData = "{\n" +
+                "        \"id\": \"2425\",\n" +
+                "        \"eventType\": \"recordInserted\",\n" +
+                "        \"subject\": \"myapp/vehicles/motorcycles\",\n" +
+                "        \"data\": {\n" +
+                "            \"attributes\": {\n" +
+                "                \"correlation-id\": \"39137f49-67d6-4001-a6aa-15521ef4f49e\",\n" +
+                "                \"data-partition-id\": \"opendes \"\n" +
+                "            },\n" +
+
+                "            \"messageId\": \"136969346945\"\n" +
+                "        },\n" +
+                "        \"dataVersion\": \"1.0\",\n" +
+                "        \"metadataVersion\": \"1\",\n" +
+                "        \"eventTime\": \"2020-08-14T18:04:12+00:00\",\n" +
+                "        \"topic\": \"/subscriptions/c99e2bf3-1777-412b-baba-d823676589c2/resourceGroups/komakkar-OSDU-RG/providers/Microsoft.EventGrid/topics/recordChanged\"\n" +
+                "    }";
+        JsonObject requestRoot = new JsonParser().parse(requestRootWithoutData).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        try{
+            // Act
+            this.sut.extractDataFromRequestBody();
+
+            // Asset
+            fail("Should Throw Exception");
+        } catch (AppException appException){
+            Assert.assertEquals(HttpStatus.BAD_REQUEST.value(), appException.getError().getCode());
+            Assert.assertEquals("data field not found", appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+
+    @Test
+    public void should_throwWhenPartitionIdIsMissing_extractDataFromRequestBody() {
+        String requestRootWithoutDataPartitionId = "{\n" +
+                "        \"id\": \"2425\",\n" +
+                "        \"eventType\": \"recordInserted\",\n" +
+                "        \"subject\": \"myapp/vehicles/motorcycles\",\n" +
+                "        \"data\": {\n" +
+                "            \"attributes\": {\n" +
+                "                \"correlation-id\": \"39137f49-67d6-4001-a6aa-15521ef4f49e\"\n" +
+                "            },\n" +
+                "            \"data\": \"W3sia2luZCI6InRlc3RraW5kIiwiaWQiOiJ0ZXN0aWQiLCJvcGVyYXRpb250eXBlIjoiY3JlYXRlIn0seyJraW5kIjoidGVzdGtpbmQyIiwiaWQiOiJ0ZXN0aWQyIiwib3BlcmF0aW9udHlwZSI6InVwZGF0ZSJ9XQ\",\n"+
+                "            \"messageId\": \"136969346945\"\n" +
+                "        },\n" +
+                "        \"dataVersion\": \"1.0\",\n" +
+                "        \"metadataVersion\": \"1\",\n" +
+                "        \"eventTime\": \"2020-08-14T18:04:12+00:00\",\n" +
+                "        \"topic\": \"/subscriptions/c99e2bf3-1777-412b-baba-d823676589c2/resourceGroups/komakkar-OSDU-RG/providers/Microsoft.EventGrid/topics/recordChanged\"\n" +
+                "    }";
+        JsonObject requestRoot = new JsonParser().parse(requestRootWithoutDataPartitionId).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        try{
+            // Act
+            this.sut.extractDataFromRequestBody();
+
+            // Asset
+            fail("Should Throw Exception");
+        } catch (AppException appException){
+            Assert.assertEquals(HttpStatus.BAD_REQUEST.value(), appException.getError().getCode());
+            //Assert.assertEquals("No tenant information from pubsub message.", appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+
+    @Test
+    public void should_returnValidData_extractDataFromRequestBody() {
+        String vaidRequestRoot = "{\n" +
+                "        \"id\": \"2425\",\n" +
+                "        \"eventType\": \"recordInserted\",\n" +
+                "        \"subject\": \"myapp/vehicles/motorcycles\",\n" +
+                "        \"data\": {\n" +
+                "            \"attributes\": {\n" +
+                "                \"correlation-id\": \"39137f49-67d6-4001-a6aa-15521ef4f49e\",\n" +
+                "                \"data-partition-id\": \"opendes \"\n" +
+                "            },\n" +
+                "            \"data\": \"W3sia2luZCI6InRlc3RraW5kIiwiaWQiOiJ0ZXN0aWQiLCJvcGVyYXRpb250eXBlIjoiY3JlYXRlIn0seyJraW5kIjoidGVzdGtpbmQyIiwiaWQiOiJ0ZXN0aWQyIiwib3BlcmF0aW9udHlwZSI6InVwZGF0ZSJ9XQ\",\n" +
+                "            \"messageId\": \"136969346945\"\n" +
+                "        },\n" +
+                "        \"dataVersion\": \"1.0\",\n" +
+                "        \"metadataVersion\": \"1\",\n" +
+                "        \"eventTime\": \"2020-08-14T18:04:12+00:00\",\n" +
+                "        \"topic\": \"/subscriptions/c99e2bf3-1777-412b-baba-d823676589c2/resourceGroups/komakkar-OSDU-RG/providers/Microsoft.EventGrid/topics/recordChanged\"\n" +
+                "    }";
+
+        String expectedData = "[{\"kind\":\"testkind\",\"id\":\"testid\",\"operationtype\":\"create\"},{\"kind\":\"testkind2\",\"id\":\"testid2\",\"operationtype\":\"update\"}]";
+
+        JsonObject requestRoot = new JsonParser().parse(vaidRequestRoot).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        // Act
+        String recievedData = this.sut.extractDataFromRequestBody();
+
+        // Asset
+        Assert.assertEquals(expectedData,recievedData);
+    }
+
+    @Test
+    public void should_returnValidAttributes_extractDataFromRequestBody() {
+        String vaidRequestRoot = "{\n" +
+                "        \"id\": \"2425\",\n" +
+                "        \"eventType\": \"recordInserted\",\n" +
+                "        \"subject\": \"myapp/vehicles/motorcycles\",\n" +
+                "        \"data\": {\n" +
+                "            \"attributes\": {\n" +
+                "                \"correlation-id\": \"39137f49-67d6-4001-a6aa-15521ef4f49e\",\n" +
+                "                \"data-partition-id\": \"opendes\"\n" +
+                "            },\n" +
+                "            \"data\": \"W3sia2luZCI6InRlc3RraW5kIiwiaWQiOiJ0ZXN0aWQiLCJvcGVyYXRpb250eXBlIjoiY3JlYXRlIn0seyJraW5kIjoidGVzdGtpbmQyIiwiaWQiOiJ0ZXN0aWQyIiwib3BlcmF0aW9udHlwZSI6InVwZGF0ZSJ9XQ\",\n" +
+                "            \"messageId\": \"136969346945\"\n" +
+                "        },\n" +
+                "        \"dataVersion\": \"1.0\",\n" +
+                "        \"metadataVersion\": \"1\",\n" +
+                "        \"eventTime\": \"2020-08-14T18:04:12+00:00\",\n" +
+                "        \"topic\": \"/subscriptions/c99e2bf3-1777-412b-baba-d823676589c2/resourceGroups/komakkar-OSDU-RG/providers/Microsoft.EventGrid/topics/recordChanged\"\n" +
+                "    }";
+        JsonObject requestRoot = new JsonParser().parse(vaidRequestRoot).getAsJsonObject();
+        when(this.requestAttributesDispatcherMock.getRequestRoot()).thenReturn(requestRoot);
+
+        // Act
+        Map<String, String> observedAttributes = this.sut.extractAttributesFromRequestBody();
+
+        // Asset
+        Assert.assertEquals(observedAttributes.get("correlation-id"),"39137f49-67d6-4001-a6aa-15521ef4f49e");
+        Assert.assertEquals(observedAttributes.get("data-partition-id"),"opendes");
+    }
+
+}
diff --git a/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/util/RequestAttributesDispatcherTest.java b/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/util/RequestAttributesDispatcherTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..1e9d8c4851de15397985da24a0c0340ed8f3f459
--- /dev/null
+++ b/provider/notification-azure/src/test/java/org/opengroup/osdu/notification/util/RequestAttributesDispatcherTest.java
@@ -0,0 +1,117 @@
+// Copyright © Microsoft Corporation
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//      http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+package org.opengroup.osdu.notification.util;
+
+import org.junit.Assert;
+import org.junit.Before;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.mockito.InjectMocks;
+import org.mockito.Mock;
+import org.mockito.Spy;
+import org.mockito.junit.MockitoJUnitRunner;
+import org.opengroup.osdu.core.common.model.http.AppException;
+import org.opengroup.osdu.notification.provider.azure.util.RequestAttributesDispatcher;
+import org.springframework.http.HttpStatus;
+
+import javax.servlet.http.HttpServletRequest;
+import java.io.BufferedReader;
+import java.io.IOException;
+import java.io.StringReader;
+
+import static org.junit.Assert.fail;
+import static org.mockito.Mockito.when;
+import static org.mockito.MockitoAnnotations.initMocks;
+
+@RunWith(MockitoJUnitRunner.class)
+public class RequestAttributesDispatcherTest {
+
+    @Mock
+    private HttpServletRequest requestMock;
+
+    @InjectMocks
+    @Spy
+    private RequestAttributesDispatcher sut;
+
+    @Before
+    public void setup() {
+        initMocks(this);
+    }
+
+    @Test
+    public void should_throwWhenSubscriptionIdMissing_getRequestSubscriptionId() {
+        when(this.requestMock.getHeader("Aeg-Subscription-Name")).thenReturn(null);
+
+        try{
+            this.sut.getRequestSubscriptionId();
+            fail("Should Throw Exception");
+        } catch (AppException appException){
+            Assert.assertEquals(HttpStatus.BAD_REQUEST.value(), appException.getError().getCode());
+            Assert.assertEquals("Subscription ID not found", appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+
+    @Test
+    public void should_throwWhenRequestRoot_getRequestRoot() throws IOException {
+        String invalidRequest = "{\n" +
+                "  \"id\": \"e094dffa-9c15-4491-8312-98c73b699908\",\n" +
+                "  \"topic\": \"/subscriptions/c99e2bf3-1777-412b-baba-d823676589c2/resourceGroups/komakkar-OSDU-RG/providers/Microsoft.EventGrid/topics/recordChanged\",\n" +
+                "  \"subject\": \"\",\n" +
+                "  \"data\": {\n" +
+                "    \"validationCode\": \"6ECAD1AE-3451-4618-9D5D-EC3DD968885D\",\n" +
+                "    \"validationUrl\": \"https://rp-westus2.eventgrid.azure.net:553/eventsubscriptions/eventgridtest/validate?id=6ECAD1AE-3451-4618-9D5D-EC3DD968885D&t=2020-08-14T11:31:24.7804184Z&apiVersion=2020-04-01-preview&token=YX71OvPJuWTTOVnhqaThLXU8gWqx6FkTkJSwoyK9%2fBE%3d\"\n" +
+                "  },\n" +
+                "  \"eventType\": \"Microsoft.EventGrid.SubscriptionValidationEvent\",\n" +
+                "  \"eventTime\": \"2020-08-14T11:31:24.7804184Z\",\n" +
+                "  \"metadataVersion\": \"1\",\n" +
+                "  \"dataVersion\": \"2\"\n" +
+                "}";
+        when(this.requestMock.getReader()).thenReturn(
+                new BufferedReader(new StringReader(invalidRequest)));
+        try {
+            this.sut.getRequestRoot();
+
+            // Assert
+            fail("Should Throw Exception");
+        } catch (AppException appException) {
+            Assert.assertEquals(HttpStatus.INTERNAL_SERVER_ERROR.value(), appException.getError().getCode());
+            Assert.assertEquals("Unable to parse request payload.", appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+
+    @Test
+    public void should_throwWhenArrayNotFound_getRequestRoot() throws IOException {
+        String invalidRequest = "[]";
+        when(this.requestMock.getReader()).thenReturn(
+                new BufferedReader(new StringReader(invalidRequest)));
+        try {
+            this.sut.getRequestRoot();
+
+            // Assert
+            fail("Should Throw Exception");
+        } catch (AppException appException) {
+            Assert.assertEquals(HttpStatus.INTERNAL_SERVER_ERROR.value(), appException.getError().getCode());
+            Assert.assertEquals("Unable to parse request payload.", appException.getError().getMessage());
+        } catch (Exception exception) {
+            fail("Should Throw AppException");
+        }
+    }
+
+}
+