Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support conversion between BigInt and binary representation #32803

Open
stevenroose opened this issue Apr 6, 2018 · 24 comments
Open

Support conversion between BigInt and binary representation #32803

stevenroose opened this issue Apr 6, 2018 · 24 comments
Labels
area-core-library SDK core library issues (core, async, ...); use area-vm or area-web for platform specific libraries. library-core library-typed-data type-enhancement A request for a change that isn't a bug

Comments

@stevenroose
Copy link

Just like bignum.BigInteger supports, please support binary conversion in BigInt as well.

More specifically, please provide the following methods:

  /// Allocates a big integer from the provided big-endian [bytes].
  BigInt.fromBytes(List<int> bytes);
  
  /// Returns a big-endian binary representation of the big integer.
  List<int> toBytes();

Possibly, both methods could take a (optional) parameters specifying the endianness, however I think big endian is by far the most common way to represent integers.

@lrhn
Copy link
Member

lrhn commented Apr 6, 2018

Little endian is efficient for many CPUs, intel and (most) ARM at least, whereas big endian is "network byte order", so often used in communications. Either makes sense, either has advantages.

I would prefer a way to write big integers into a ByteBuffer or Uint8List, instead of having to create an intermediate list (although the returned buffer of toBytes could be a view of the internal buffer of the BigInt if it has the correct endianness). In that case we would just do what all the other ByteBuffer methods do and allow an Endian argument, that is, just support both endianesses and default to Endian.host.

@stevenroose
Copy link
Author

stevenroose commented Apr 6, 2018

As a reference, Java's BigInteger class has a default constructor that takes a big-endian byte array and a toByteArray() method that returns big-endian (two complement) bytes.

The BigInteger class from the bignum package also works like that.

I'm currently migrating the Pointy Castle code to BigInt and we also quite heavily rely on that functionality.

@azenla
Copy link
Contributor

azenla commented Apr 6, 2018

@lrhn I think that adding it to dart:typed_data like a normal fixed size integer wouldn't work, as BigInt is almost certainly implemented in a variable size.

So, if I needed to copy a bigint into a uint8list, I wouldn't have a good way of knowing how big to make the initial buffer.

@stevenroose
Copy link
Author

@lrhn Also, since BigInt is in core, it needs to return List<int>, but under the hood, it will most certainly be a Uint8List, since that makes the most sense.

@lrhn
Copy link
Member

lrhn commented Apr 8, 2018

BigInt has bitLength, so you do know its actual size. That would allow you to serialize it into, say, a Uint8Buffer if you ensure the capacity first. I'd still want that functionality, either on BigInt or on Unit8List/ByteData since it would be useful for serializing a BigInt. Creating an intermediate list is wasteful (but again, BigInt is immutable, so the list could be a view on the internal structure if that structure has a useful ordering, probably little endian).

Core methods can return typed data lists, and be typed as such, we just try to avoid it if it isn't necessary. I'd probably prefer adding setBigInt to ByteData rather than writeToByteData on BigInt.

@stevenroose
Copy link
Author

@lrhn What is the roadmap for this? I agree that for efficient serialization etc, a method in ByteData might make sense, but for other applications, I think just having a toBytes (and equivalent from) is perfectly fine.

@a-siva a-siva added the area-core-library SDK core library issues (core, async, ...); use area-vm or area-web for platform specific libraries. label Apr 12, 2018
@stevenroose
Copy link
Author

Any update on this??

@lrhn
Copy link
Member

lrhn commented May 8, 2018

We are very busy doing the already planned Dart 2 features, so I don't think it's likely that a feature like this will make it into Dart 2.0. (It's kind-of trivial once the design is nailed down, but so are most of the other things we are currently doing).

For now, I'll recommend just using helper functions like:

BigInt readBytes(Uint8List bytes) {
  BigInt read(int start, int end) {
    if (end - start <= 4) {
      int result = 0;
      for (int i = end - 1; i >= start; i--) {
        result = result * 256 + bytes[i];
      }
      return new BigInt.from(result);
    }
    int mid = start + ((end - start) >> 1);
    var result = read(start, mid) + read(mid, end) * (BigInt.one << ((mid - start) * 8));
    return result;
  }
  return read(0, bytes.length);
}

Uint8List writeBigInt(BigInt number) {
  // Not handling negative numbers. Decide how you want to do that.
  int bytes = (number.bitLength + 7) >> 3;
  var b256 = new BigInt.from(256);
  var result = new Uint8List(bytes);
  for (int i = 0; i < bytes; i++) {
    result[i] = number.remainder(b256).toInt();
    number = number >> 8;
  }
  return result;
}

(There are lots of opportunities for optimizations, obviously).

@stevenroose
Copy link
Author

(You made a typo on that long line.)

Yeah that was my alternative, but I was kinda trying to avoid that. But yeah since it won't land 2.0, I'll probably do this instead.

@sergiogragera
Copy link

sergiogragera commented Sep 20, 2018

Does anyone know of any method to convert uin8list to core.BigInt? How have you corrected the typo? Solution on https://github.com/stevenroose/dart-cryptoutils/pull/5/files works for me (changing some data type) but I don't know if is the correct solution. I want a method like Java BigInteger.

@lrhn
Copy link
Member

lrhn commented Sep 24, 2018

Typo has been corrected (line was truncated when cut-n-pasted from a console).

@haarts
Copy link

haarts commented Oct 1, 2018

Not sure if this belongs here but I found it helpful; the authors of PointyCastle include a utility to do the conversion: https://github.com/PointyCastle/pointycastle/blob/master/lib/src/utils.dart (actually @stevenroose made that change). If you are using PointyCastle anyway I think that is a good alternative.

@stevenroose
Copy link
Author

Any update on this?

@kevmoo
Copy link
Member

kevmoo commented Jan 9, 2019

@lrhn – could we add a constructor on ByteBuffer to take BigInt? Require it be non-negative, etc.

@kevmoo
Copy link
Member

kevmoo commented Jan 19, 2019

@lrhn @mraleph thoughts here?

@rakudrama
Copy link
Member

What is expected of the 'byte view' of negative BigInt values?
It is not specified if the internal representation is two's complement or sign-magnitude and that would affect any kind of 'view'.

We also need to consider if the operations can be implemented efficiently for JavaScript BigInt values, since we will eventually want to use JavaScript BigInt when compiling for the browser. Tree-structured algorithms like readBytes sketched by @lrhn are probably the way to go for big values, but that sketch could be improved considerably. It would be a big improvement for the compiler to recognize compound operations, e.g. (a + (b << c)) that can be implemented more efficiently without the intermediate values.

@rakudrama
Copy link
Member

@stevenroose How are the requested APIs used? i.e. why are they useful? Is there a standard that requires the format?

@lrhn
Copy link
Member

lrhn commented Mar 11, 2021

We now plan to seal typed_data classes (#45115), which also opens up the ability to add members on the classes without breaking anyone. That makes it more likely that we'll actually be able to add a setBigInt method on byteData.
It's still an issue that the BigInt class itself isn't sealed, so we can't assume anything about the underlying representation.

For that reason, a writeBytes(Uint8List target, int start, {Endian? endian) on BigInt would be better, but a breaking change to add.

@rakudrama
Copy link
Member

@lrhn Should BigInt be sealed too?

That would preserve a path for BigInt constants, BigInt literals, implementation in dart2js via JavaScript BigInt values, and interoperability with BigInt64Array.

@lrhn
Copy link
Member

lrhn commented Mar 15, 2021

I think it would be a good idea to seal BigInt.

Effectively, we've already assumed that it is sealed because we start all the instance methods with _ensureSystemBigInt(other) anyway, which throws if passed anything other than the system implementation of BigInt.

It'd be a different request, though. I'd vote for it.

denizt added a commit to denizt/pc-dart that referenced this issue May 14, 2021
A direct port from bouncy-castle to add SRP support to
pointy-castle. For details on the protocol visit
https://en.wikipedia.org/wiki/Secure_Remote_Password_protocol

There is code duplication in encode/decode BigInt due to
dart not yet supporting conversion natively. Tracked on
dart-lang/sdk#32803

When using the protocol, we use the following parameters to
improve security and efficiency, please do your research
as the protocol itself will not protect you from mistakes.

Digest: "SHA-256"
SecureRandom: "AES/CTR/AUTO-SEED-PRNG";
N,g: Generate own big prime(4096)

We also recommend using a KDF, such as scrypt to secure the
password, hence the verifier.
@mark-dropbear
Copy link

Figured I would leave this here in case anybody else came across it in the future but I ran into a scenario where I needed this recently here: https://github.com/dropbear-software/todart/blob/main/common/lib/src/types/cloud_resource_identity.dart

Here is how I solved it thanks to some help from others in the community Gitter chat

/// Converts a [Uint8List] byte buffer into a [BigInt]
static BigInt _convertBytesToBigInt(Uint8List bytes) {
  BigInt result = BigInt.zero;

  for (final byte in bytes) {
    // reading in big-endian, so we essentially concat the new byte to the end
    result = (result << 8) | BigInt.from(byte);
  }
  return result;
}

@kevmoo
Copy link
Member

kevmoo commented Nov 8, 2021

@mark-dropbear – make that an extension method and we have 🥇 !!!

@mark-dropbear
Copy link

mark-dropbear commented Nov 9, 2021

Thanks @kevmoo I will take a look at that later on, have never used extension methods before but I am excited to check them out. By the way, that project I referenced there is most certainly not yet in a state that has a whole lot worth talking about just yet but is heavily inspired by Knarly Vote (There are a few big differences like gRPC rather than cloud functions for example).

However, the entire concept is to kind of just use that code base as a playground to explore what fullstack dart could look like and especially while trying to do things the "Google AIP way".

I'm intentionally trying to keep the application itself relatively simple (it's just a Todo list kind of app) so that I have something that at least is going to have to bump into "real life problems" but easy enough that I can just focus on the patterns and see where the messy parts are.

I was going to ping you when I had something there worth sharing because I figured that both you and a few others might get a kick out of it.

@ssddOnTop
Copy link

ssddOnTop commented Aug 26, 2022

Hey I was recently working on a project for e2ee for multiple languages and I stumbled upon this problem, I tried @mark-dropbear and @lrhn 's solutions but they were a bit incomplete, the method _convertBytesToBigInt() assumes big-endianness whereas writeBigInt() writes in little-endian so I fixed them.

  static BigInt readBytes(Uint8List bytes) {
    BigInt result = BigInt.zero;

    for (final byte in bytes) {
      // reading in big-endian, so we essentially concat the new byte to the end
      result = (result << 8) | BigInt.from(byte & 0xff);
    }
    return result;
  }

  static Uint8List writeBigInt(BigInt number) {
    // Not handling negative numbers. Decide how you want to do that.
    int bytes = (number.bitLength + 7) >> 3;
    var b256 = BigInt.from(256);
    var result = Uint8List(bytes);
    for (int i = 0; i < bytes; i++) {
      result[bytes - 1 - i] = number.remainder(b256).toInt();
      number = number >> 8;
    }
    return result;
  }

now I'm able to use the same methods for multiple languages without any problems.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area-core-library SDK core library issues (core, async, ...); use area-vm or area-web for platform specific libraries. library-core library-typed-data type-enhancement A request for a change that isn't a bug
Projects
None yet
Development

No branches or pull requests

10 participants