All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.hyperfoil.core.handlers.SearchHandler Maven / Gradle / Ivy

There is a newer version: 0.27.1
Show newest version
package io.hyperfoil.core.handlers;

import java.nio.charset.StandardCharsets;

import io.hyperfoil.api.processor.Processor;
import io.netty.buffer.ByteBuf;
import io.hyperfoil.api.session.Session;
import io.hyperfoil.api.session.ResourceUtilizer;

/**
 * Simple pattern (no regexp) search based on Rabin-Karp algorithm.
 * Does not handle the intricacies of UTF-8 mapping same strings to different bytes.
 */
public class SearchHandler implements Processor, ResourceUtilizer, Session.ResourceKey {
   private final byte[] begin, end;
   private final int beginHash, endHash;
   private final int beginCoef, endCoef;
   private Processor processor;

   public SearchHandler(String begin, String end, Processor processor) {
      this.begin = begin.getBytes(StandardCharsets.UTF_8);
      this.end = end.getBytes(StandardCharsets.UTF_8);
      this.beginHash = BaseSearchContext.computeHash(this.begin);
      this.beginCoef = BaseSearchContext.computeCoef(this.begin.length);
      this.endHash = BaseSearchContext.computeHash(this.end);
      this.endCoef = BaseSearchContext.computeCoef(this.end.length);
      this.processor = processor;
   }

   @Override
   public void before(Session session) {
      Context ctx = session.getResource(this);
      ctx.reset();
      processor.before(session);
   }

   @Override
   public void process(Session session, ByteBuf data, final int offset, int length, boolean isLast) {
      Context ctx = session.getResource(this);
      ctx.add(data, offset, length);
      int endIndex = offset + length;
      int index = ctx.initHash(offset, ctx.lookupText.length);
      while (ctx.test(index)) {
         if (ctx.lookupText == end) {
            fireProcessor(ctx, session, index);
         } else {
            ctx.mark(index);
         }
         ctx.swap();
         index = ctx.initHash(index, ctx.lookupText.length);
      }
      while (index < endIndex) {
         ctx.advance(data.getByte(index++), ctx.lookupCoef, index, ctx.lookupText.length + 1);
         while (ctx.test(index)) {
            if (ctx.lookupText == end) {
               fireProcessor(ctx, session, index);
            } else {
               ctx.mark(index);
            }
            ctx.swap();
            index = ctx.initHash(index, ctx.lookupText.length);
         }
      }
   }

   private void fireProcessor(Context ctx, Session session, int index) {
      int endPart = ctx.currentPart;
      int endPos = index - end.length;
      while (endPos < 0) {
         endPart--;
         if (endPart < 0) {
            endPart = 0;
            endPos = 0;
         } else {
            endPos += ctx.endIndices[endPart];
         }
      }
      while (ctx.markPart < endPart) {
         ByteBuf data = ctx.parts[ctx.markPart];
         // if the begin ends with part, we'll skip the 0-length process call
         int length = ctx.endIndices[ctx.markPart] - ctx.markPos;
         if (length > 0) {
            processor.process(session, data, ctx.markPos, length, false);
         }
         ctx.markPos = 0;
         ctx.markPart++;
      }
      processor.process(session, ctx.parts[endPart], ctx.markPos, endPos - ctx.markPos, true);
   }

   @Override
   public void after(Session session) {
      Context ctx = session.getResource(this);
      // release buffers
      ctx.reset();
      processor.after(session);
   }

   @Override
   public void reserve(Session session) {
      session.declareResource(this, new Context());
   }

   class Context extends BaseSearchContext {
      byte[] lookupText;
      int lookupHash;
      int lookupCoef;
      int markPart = -1;
      int markPos = -1;

      void swap() {
         currentHash = 0;
         hashedBytes = 0;
         if (lookupText == end) {
            lookupText = begin;
            lookupHash = beginHash;
            lookupCoef = beginCoef;
         } else {
            lookupText = end;
            lookupHash = endHash;
            lookupCoef = endCoef;
         }
      }

      boolean test(int index) {
         if (currentHash == lookupHash) {
            for (int i = 0; i < lookupText.length; ++i) {
               if (lookupText[i] != byteRelative(index, lookupText.length - i)) {
                  return false;
               }
            }
            return true;
         }
         return false;
      }

      @Override
      void shiftParts() {
         super.shiftParts();
         --markPart;
         if (markPart < 0) {
            markPart = 0;
            markPos = 0;
         }
      }

      @Override
      void reset() {
         super.reset();
         lookupText = begin;
         lookupHash = beginHash;
         lookupCoef = beginCoef;
         markPart = -1;
         markPos = -1;
      }

      void mark(int index) {
         markPart = currentPart;
         markPos = index;
      }
   }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy