| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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.apache.nifi.controller.swap; |
| |
| import org.apache.nifi.controller.queue.FlowFileQueue; |
| import org.apache.nifi.controller.repository.FlowFileRecord; |
| import org.apache.nifi.controller.repository.SwapContents; |
| import org.apache.nifi.controller.repository.SwapSummary; |
| import org.apache.nifi.controller.repository.claim.ContentClaim; |
| import org.apache.nifi.controller.repository.claim.ResourceClaim; |
| import org.apache.nifi.controller.repository.claim.ResourceClaimManager; |
| import org.apache.nifi.controller.repository.claim.StandardResourceClaimManager; |
| import org.apache.nifi.flowfile.FlowFile; |
| import org.apache.nifi.stream.io.NullOutputStream; |
| import org.junit.jupiter.api.BeforeEach; |
| import org.junit.jupiter.api.Disabled; |
| import org.junit.jupiter.api.Test; |
| import org.mockito.Mockito; |
| |
| import java.io.BufferedInputStream; |
| import java.io.BufferedOutputStream; |
| import java.io.DataInputStream; |
| import java.io.File; |
| import java.io.FileInputStream; |
| import java.io.FileOutputStream; |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.io.OutputStream; |
| import java.nio.file.Files; |
| import java.util.ArrayList; |
| import java.util.HashMap; |
| import java.util.HashSet; |
| import java.util.List; |
| import java.util.Map; |
| import java.util.Set; |
| import java.util.concurrent.TimeUnit; |
| import java.util.stream.Collectors; |
| |
| import static org.junit.jupiter.api.Assertions.assertEquals; |
| import static org.junit.jupiter.api.Assertions.assertFalse; |
| |
| public class TestSchemaSwapSerializerDeserializer { |
| |
| @BeforeEach |
| public void setup() { |
| MockFlowFile.resetIdGenerator(); |
| } |
| |
| @Test |
| public void testRoundTripSerializeDeserializeSummary() throws IOException { |
| final ResourceClaimManager resourceClaimManager = new StandardResourceClaimManager(); |
| final ResourceClaim firstResourceClaim = resourceClaimManager.newResourceClaim("container", "section", "id", true, false); |
| resourceClaimManager.incrementClaimantCount(firstResourceClaim); |
| |
| final List<FlowFileRecord> toSwap = new ArrayList<>(10000); |
| final Map<String, String> attrs = new HashMap<>(); |
| long size = 0L; |
| final ContentClaim firstClaim = MockFlowFile.createContentClaim("id", resourceClaimManager); |
| for (int i = 0; i < 10000; i++) { |
| attrs.put("i", String.valueOf(i)); |
| final FlowFileRecord ff = i < 2 ? new MockFlowFile(attrs, i, firstClaim) : new MockFlowFile(attrs, i, resourceClaimManager); |
| toSwap.add(ff); |
| size += i; |
| } |
| |
| final FlowFileQueue flowFileQueue = Mockito.mock(FlowFileQueue.class); |
| Mockito.when(flowFileQueue.getIdentifier()).thenReturn("87bb99fe-412c-49f6-a441-d1b0af4e20b4"); |
| |
| final String swapLocation = "target/testRoundTrip.swap"; |
| final File swapFile = new File(swapLocation); |
| Files.deleteIfExists(swapFile.toPath()); |
| |
| final SwapSerializer serializer = new SchemaSwapSerializer(); |
| try (final FileOutputStream fos = new FileOutputStream(swapFile)) { |
| serializer.serializeFlowFiles(toSwap, flowFileQueue, swapLocation, fos); |
| } |
| |
| final SwapDeserializer deserializer = new SchemaSwapDeserializer(); |
| final SwapSummary swapSummary; |
| try (final FileInputStream fis = new FileInputStream(swapFile); |
| final DataInputStream dis = new DataInputStream(fis)) { |
| |
| swapSummary = deserializer.getSwapSummary(dis, swapLocation, resourceClaimManager); |
| } |
| |
| assertEquals(10000, swapSummary.getQueueSize().getObjectCount()); |
| assertEquals(size, swapSummary.getQueueSize().getByteCount()); |
| assertEquals(9999, swapSummary.getMaxFlowFileId().intValue()); |
| |
| final List<ResourceClaim> resourceClaims = swapSummary.getResourceClaims(); |
| assertEquals(10000, resourceClaims.size()); |
| assertFalse(resourceClaims.stream().anyMatch(claim -> claim == null)); |
| assertEquals(2, resourceClaims.stream().filter(claim -> claim.getId().equals("id")).collect(Collectors.counting()).intValue()); |
| |
| final Set<ResourceClaim> uniqueClaims = new HashSet<>(resourceClaims); |
| assertEquals(9999, uniqueClaims.size()); |
| |
| assertEquals((Long) toSwap.stream().mapToLong(FlowFile::getLastQueueDate).sum(), swapSummary.getTotalLastQueueDate()); |
| assertEquals((Long) toSwap.stream().mapToLong(FlowFile::getLastQueueDate).min().getAsLong(), swapSummary.getMinLastQueueDate()); |
| } |
| |
| @Test |
| public void testRoundTripSerializeDeserializeFullSwapFile() throws IOException, InterruptedException { |
| final ResourceClaimManager resourceClaimManager = new StandardResourceClaimManager(); |
| |
| final List<FlowFileRecord> toSwap = new ArrayList<>(10000); |
| final Map<String, String> attrs = new HashMap<>(); |
| long size = 0L; |
| for (int i = 0; i < 10000; i++) { |
| attrs.put("i", String.valueOf(i)); |
| final FlowFileRecord ff = new MockFlowFile(attrs, i, resourceClaimManager); |
| toSwap.add(ff); |
| size += i; |
| } |
| |
| final FlowFileQueue flowFileQueue = Mockito.mock(FlowFileQueue.class); |
| Mockito.when(flowFileQueue.getIdentifier()).thenReturn("87bb99fe-412c-49f6-a441-d1b0af4e20b4"); |
| |
| final String swapLocation = "target/testRoundTrip.swap"; |
| final File swapFile = new File(swapLocation); |
| Files.deleteIfExists(swapFile.toPath()); |
| |
| final SwapSerializer serializer = new SchemaSwapSerializer(); |
| try (final OutputStream fos = new FileOutputStream(swapFile); |
| final OutputStream out = new BufferedOutputStream(fos)) { |
| serializer.serializeFlowFiles(toSwap, flowFileQueue, swapLocation, out); |
| } |
| |
| final SwapContents contents; |
| final SwapDeserializer deserializer = new SchemaSwapDeserializer(); |
| try (final FileInputStream fis = new FileInputStream(swapFile); |
| final InputStream bufferedIn = new BufferedInputStream(fis); |
| final DataInputStream dis = new DataInputStream(bufferedIn)) { |
| |
| contents = deserializer.deserializeFlowFiles(dis, swapLocation, flowFileQueue, resourceClaimManager); |
| } |
| |
| final SwapSummary swapSummary = contents.getSummary(); |
| assertEquals(10000, swapSummary.getQueueSize().getObjectCount()); |
| assertEquals(size, swapSummary.getQueueSize().getByteCount()); |
| assertEquals(9999, swapSummary.getMaxFlowFileId().intValue()); |
| |
| assertEquals(10000, contents.getFlowFiles().size()); |
| |
| int counter = 0; |
| for (final FlowFileRecord flowFile : contents.getFlowFiles()) { |
| final int i = counter++; |
| assertEquals(String.valueOf(i), flowFile.getAttribute("i")); |
| assertEquals(i, flowFile.getSize()); |
| } |
| } |
| |
| @Test |
| @Disabled("For manual testing, in order to ensure that changes do not negatively impact performance") |
| public void testWritePerformance() throws IOException, InterruptedException { |
| final ResourceClaimManager resourceClaimManager = new StandardResourceClaimManager(); |
| |
| final List<FlowFileRecord> toSwap = new ArrayList<>(10000); |
| final Map<String, String> attrs = new HashMap<>(); |
| for (int i = 0; i < 10000; i++) { |
| attrs.put("i", String.valueOf(i)); |
| final FlowFileRecord ff = new MockFlowFile(attrs, i, resourceClaimManager); |
| toSwap.add(ff); |
| } |
| |
| final FlowFileQueue flowFileQueue = Mockito.mock(FlowFileQueue.class); |
| Mockito.when(flowFileQueue.getIdentifier()).thenReturn("87bb99fe-412c-49f6-a441-d1b0af4e20b4"); |
| |
| final String swapLocation = "target/testRoundTrip.swap"; |
| |
| final int iterations = 1000; |
| |
| final long start = System.nanoTime(); |
| final SwapSerializer serializer = new SchemaSwapSerializer(); |
| for (int i = 0; i < iterations; i++) { |
| try (final OutputStream out = new NullOutputStream()) { |
| serializer.serializeFlowFiles(toSwap, flowFileQueue, swapLocation, out); |
| } |
| } |
| |
| final long millis = TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - start); |
| System.out.println("Wrote " + iterations + " Swap Files in " + millis + " millis"); |
| } |
| } |