package org.msgpack.jruby;

import org.jruby.Ruby; import org.jruby.RubyModule; import org.jruby.RubyClass; import org.jruby.RubyString; import org.jruby.RubyNil; import org.jruby.RubyBoolean; import org.jruby.RubyHash; import org.jruby.runtime.load.Library; import org.jruby.runtime.builtin.IRubyObject; import org.jruby.runtime.ThreadContext; import org.jruby.runtime.Block; import org.jruby.runtime.Visibility; import org.jruby.anno.JRubyModule; import org.jruby.anno.JRubyMethod; import org.jruby.internal.runtime.methods.CallConfiguration; import org.jruby.internal.runtime.methods.DynamicMethod;

public class MessagePackLibrary implements Library {

public void load(Ruby runtime, boolean wrap) {
  RubyModule msgpackModule = runtime.defineModule("MessagePack");
  msgpackModule.defineAnnotatedMethods(MessagePackModule.class);
  RubyClass standardErrorClass = runtime.getStandardError();
  RubyClass unpackErrorClass = msgpackModule.defineClassUnder("UnpackError", standardErrorClass, standardErrorClass.getAllocator());
  RubyClass underflowErrorClass = msgpackModule.defineClassUnder("UnderflowError", unpackErrorClass, unpackErrorClass.getAllocator());
  RubyClass extensionValueClass = msgpackModule.defineClassUnder("ExtensionValue", runtime.getObject(), new ExtensionValue.ExtensionValueAllocator());
  extensionValueClass.defineAnnotatedMethods(ExtensionValue.class);
  RubyClass packerClass = msgpackModule.defineClassUnder("Packer", runtime.getObject(), new Packer.PackerAllocator());
  packerClass.defineAnnotatedMethods(Packer.class);
  RubyClass unpackerClass = msgpackModule.defineClassUnder("Unpacker", runtime.getObject(), new Unpacker.UnpackerAllocator());
  unpackerClass.defineAnnotatedMethods(Unpacker.class);
  RubyClass bufferClass = msgpackModule.defineClassUnder("Buffer", runtime.getObject(), new Buffer.BufferAllocator());
  bufferClass.defineAnnotatedMethods(Buffer.class);
  installCoreExtensions(runtime);
}

private void installCoreExtensions(Ruby runtime) {
  installCoreExtensions(
    runtime,
    runtime.getNilClass(),
    runtime.getTrueClass(),
    runtime.getFalseClass(),
    runtime.getFixnum(),
    runtime.getBignum(),
    runtime.getFloat(),
    runtime.getString(),
    runtime.getArray(),
    runtime.getHash(),
    runtime.getSymbol()
  );
}

private void installCoreExtensions(Ruby runtime, RubyClass... classes) {
  for (RubyClass cls : classes) {
    cls.addMethod("to_msgpack", createToMsgpackMethod(runtime, cls));
  }
}

private DynamicMethod createToMsgpackMethod(final Ruby runtime, RubyClass cls) {
  return new DynamicMethod(cls, Visibility.PUBLIC, CallConfiguration.FrameNoneScopeNone) {
    @Override
    public IRubyObject call(ThreadContext context, IRubyObject recv, RubyModule clazz, String name, IRubyObject[] args, Block block) {
      IRubyObject[] allArgs = new IRubyObject[1 + args.length];
      allArgs[0] = recv;
      System.arraycopy(args, 0, allArgs, 1, args.length);
      return MessagePackModule.pack(runtime.getCurrentContext(), null, allArgs);
    }

    @Override
    public DynamicMethod dup() {
      return this;
    }
  };
}

@JRubyModule(name = "MessagePack")
public static class MessagePackModule {
  @JRubyMethod(module = true, required = 1, optional = 1, alias = {"dump"})
  public static IRubyObject pack(ThreadContext ctx, IRubyObject recv, IRubyObject[] args) {
    IRubyObject[] extraArgs = null;
    if (args.length == 0) {
      extraArgs = new IRubyObject[] {};
    } else {
      extraArgs = new IRubyObject[args.length - 1];
      System.arraycopy(args, 1, extraArgs, 0, args.length - 1);
    }
    Packer packer = new Packer(ctx.getRuntime(), ctx.getRuntime().getModule("MessagePack").getClass("Packer"));
    packer.initialize(ctx, extraArgs);
    packer.write(ctx, args[0]);
    return packer.toS(ctx);
  }

  @JRubyMethod(module = true, required = 1, optional = 1, alias = {"load"})
  public static IRubyObject unpack(ThreadContext ctx, IRubyObject recv, IRubyObject[] args) {
    Decoder decoder = new Decoder(ctx.getRuntime(), args[0].asString().getBytes());
    if (args.length > 1 && !args[args.length - 1].isNil()) {
      RubyHash hash = args[args.length - 1].convertToHash();
      IRubyObject symbolizeKeys = hash.fastARef(ctx.getRuntime().newSymbol("symbolize_keys"));
      decoder.symbolizeKeys(symbolizeKeys != null && symbolizeKeys.isTrue());
    }
    return decoder.next();
  }
}

}