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

Time locked transaction outputs #25

Open
ignopeverell opened this issue Dec 23, 2016 · 24 comments
Open

Time locked transaction outputs #25

ignopeverell opened this issue Dec 23, 2016 · 24 comments

Comments

@ignopeverell
Copy link
Contributor

How do we support them? Would likely require another output field and a signature. Maybe add to the transaction proof signature.

This would be great to have for higher level contract supports like payment channels, lightning, etc.

@apoelstra
Copy link

This belongs in the transaction signature. Yeah, it'd have to be encoded in another field.

@ignopeverell
Copy link
Contributor Author

Given that we already sign the fee, it'd either have to get composed or added as another signature.

@antiochp
Copy link
Member

antiochp commented Oct 9, 2017

Been thinking a little about this.

We currently sign the fee on the transaction itself - so presumably we could include a height or time in the message being signed.

We do the following to create the message (with just the fee currently) -

fn u64_to_32bytes(n: u64) -> [u8; 32] {
	let mut bytes = [0; 32];
	BigEndian::write_u64(&mut bytes[24..32], n);
	bytes

So there is space in there to include the timelock info.

But - this limits the timelock to the entire transaction, so all outputs would be affected by this, change outputs as well.

Is this an acceptable approach?
Ideally the timelock would be per-output but we have no way of doing this in MW - we don't sign anything limited to just a single output.

I guess the general approach would be to make the necessary transactions within the wallet to build a single output of the right amount and then send the timelocked transaction creating a single timelocked output (and avoid ending up with a potentially large change output also encumbered with a timelock).

@antiochp
Copy link
Member

antiochp commented Oct 9, 2017

I also saw some mention on the ML of keeping the TxKernel as small as possible and aiming to avoid needing to persist a timelock on every TxKernel.

We have the features bitflags in there - so I'm thinking we could use one of the bits in features to flag timelocked transactions - only serialize/deserialize the timelock if present.

This would mean the majority of TxKernels would remain the same size (assuming most are not encumbered by timelocks).

@apoelstra
Copy link

Yes, the only thing MW can do that I'm aware of is put a timelock on a kernel, which effectively timelocks the whole transaction.

I'd prefer that every kernel have a timelock to make "real" ones inconspicuous, for privacy's sake. But it's not a strong preference.

@antiochp
Copy link
Member

antiochp commented Oct 9, 2017

@apoelstra I thought I saw you recommend on the ML somewhere to avoid putting a timelock on every kernel to minimize the storage cost?
But yeah I agree it makes sense to have it on all of them to keep them all similar.

Except - is it not going to be trivially easy to identify no-op timelocks or very short lock periods by just comparing the kernels to the height or timestamp of the blocks they originate from?
Say an output with a lock_height of 100 originating in a block of height 100 (so no effective locktime) is going to be readily distinguishable from an output in the same block with a lock_height of 600 (500 blocks later).

@apoelstra
Copy link

There isn't any way to do relative locktimes (that I know of) -- the way kernel locktimes work is that they make that kernel invalid until the specified block is up.

Relative locktimes would require validators learn when outputs were spent relative to when they were created, which isn't possible in general in MW because later validators never learn about old spent outputs. There may still be value in supporting this because typical relative locktimes are quite short-lived, but these locktimes would have a much weaker form of security than the rest of the system. They would need to be associated with outputs rather than kernels.

Yes, I did comment on how I wanted to save space, but Igno (I think) convinced me that "only four bytes" per kernel was fine :).

@antiochp
Copy link
Member

Oh interesting. I had not fully thought through the output validation vs kernel validation.
The way we're doing locktimes on coinbase outputs is based on validating the output, not the kernel.
I need to go read through that again and see if I can understand how to apply what you're saying here to coinbase locktimes.

@apoelstra
Copy link

Even with coinbase outputs, historical validators won't be able to validate that the funds were unmoved until maturity. I think this is acceptable, the maturity rule is only about reorg safety and to "exploit this" you have to rewrite way more blocks than you would to just replace mature coinbases in the first place.

I hadn't noticed this before. Interesting.

@antiochp
Copy link
Member

Initial PR up for discussion (based on conversation above) - #167

@ignopeverell
Copy link
Contributor Author

Another idea. Assume we have hashed switch commitments as proposed by Andrew in [1]:

vH + rG, sha256(rJ)

By default we reuse the r blinding factor. How about time locked transactions use the block height h instead? On spend, the input would have to reveal h.

[1] https://lists.launchpad.net/mimblewimble/msg00165.html

@ignopeverell
Copy link
Contributor Author

As discussed on Gitter, my proposal was for time-lock outputs, whereas the previous discussion mostly revolved around time-locked transactions (that can't be included in a block before h).

@gosthell
Copy link
Contributor

For time-locked output, couldn't we introduce a new NUM base point K such that an output commits to vH + rG + hK where h is the height at which the output becomes spendable. When someone wants to spend the output, the h value has to be revealed so that the verifier can subtract hK from the excess, and check if the current block height is higher than h.

@apoelstra
Copy link

@melaurent if the locktime height should be hidden until it is used (which TBH doesn't seem super useful to me), it is more verifier-efficient to just hide it behind a hash and reveal it explicitly. Then no EC math needs to be done.

Further, with your suggestion n would need to be stored forever so that historical validators could learn what multiple of K to subtract from the utxo set. By avoiding EC math, it's possible to design things so that historical validators never learn about locktimes. Strictly speaking this weakens the security model for locktimes, but I think the resulting security model is exactly what we want for locktimes because they're so short-lived. That is, online validators ensure they are not validated until they are long-buried and long-expired, at which point violating them would require rewriting so many blocks that the violation was unnecessary anyway, and then nobody looks at them again.

@apoelstra
Copy link

Oops! I realize suddenly that there is a miner-collusion vulnerability with outputs that have relative locktimes. If somebody spends these before the tx is confirmed (so clearly way before the locktime is up), a miner can include both transactions. The timelocked output will get cut-through and nobody will know that any rule was violated.

@antiochp
Copy link
Member

Oh - so its not valid to put the tx in the pool (this can be consensus enforced).
But the miner can skip the pool entirely and include the tx in the block?

@ignopeverell
Copy link
Contributor Author

A miner can include any transaction they want without telling anyone else until it's in the block, yes.

Regarding output lock times, as those are in the chain, assumingly whoever relies on them would wait until they're mined and detected in the chain before proceeding with the contract? Also right now we don't support multi-kernel transactions.

@apoelstra
Copy link

@antiochp I'm saying that these outputs are not secure at all until they've been deeply confirmed.

@ignopeverell I'm actually not sure what application unconditionally locked outputs have, so I don't know how protocols using them would likely work. But this seems like a pretty serious footgun.

@antiochp
Copy link
Member

antiochp commented Oct 30, 2017

(this is all assuming we can solve the issue with miner collusion and zero confirmations).

Switch commit proposal -

vH + rG, sha256(rJ)

The range proof signs the extra switch commitment rJ so this cannot be modified.

@ignopeverell proposal above to use the switch commit to hide the lock_height h (by using it as the blinding factor in the switch commitment) -

vH + rG, sha256(hJ)

What if we used blake2 instead of sha256 as the hashing function?
Edit: I see we are actually using blake2 in the code (but without the secret key).

We could then pass h in as the secret key to the blake2 function.

vH + rG, blake2(h, rJ)

We could then preserve reuse of r across both the commitment and the switch commitment (assuming this is desirable).

To spend you would need to reveal h and rJ.

I'm sure I'm missing something really basic here as to why this falls down. Thoughts?

@ignopeverell ignopeverell modified the milestones: Testnet, Mainnet Oct 31, 2017
@ignopeverell
Copy link
Contributor Author

ignopeverell commented Nov 2, 2017

Just to circle back, miner collusion is indeed an issue in the general case. And actually even without miner collusion, if we start allowing multi-kernel transactions (we don't at the moment), it will become even more of a problem.

However in the particular case of time-locking miner rewards (which is what I believe you're after here), I don't see any immediate issue with this scheme. It would also be a nice way to try out switch commitment hash pre-image reveals.

@antiochp
Copy link
Member

antiochp commented Nov 2, 2017

Multi-kernel transactions will come into play when we start using Schnorr signatures or is this something else? My understanding is we will have multiple kernels when multiple parties are involved in building parts of the tx independently?

Could you explain how this is involved as I don't think I understand the impact here?
Thanks!

@ignopeverell
Copy link
Contributor Author

Schnorr signatures are aggregates (they've actually been rebranded as aggregate signatures or aggsigs) so that would still only generate a single kernel. In short, each party generate a partial signature and all of them can then be aggregated into a single final one.

Multiple kernels come into play for the current issue because if you want to do cut-through between 2 transactions, generating a single one afterward, you do need to be able to have multiple kernels. Otherwise the resulting "merged" transaction would not sum correctly with the kernel. Does that make more sense?

@antiochp
Copy link
Member

antiochp commented Nov 3, 2017

Yeah ok I see how we'd end up with a tx with multiple kernels.

But I don't see how this affects outputs?
Is the problem that if we allow txs to be merged (by allowing multiple kernels) then anyone can effectively merge them and at that point we can no longer enforce any of this via consensus rules because inputs and outputs can effectively be removed before they are added to the chain?

Just thinking through this some more - can any two txs effectively be merged like this?
I was assuming they could only be merged if one spent an output from the other - but that's not actually the case. We wouldn't gain much by merging two completely unrelated txns - but it could still happen?
I think I answered my own question...

@antiochp
Copy link
Member

antiochp commented Nov 3, 2017

Sort of related - I know you mentioned coin-join previously.
I thought coin-join was only effective if all the outputs were of the same value, a bunch of interchangeable outputs with the same denomination.
But this is not applicable to Grin as the amounts are hidden? So can any outputs put together end up doing something similar (but we can effectively ignore amounts and treat all outputs as the same for this purpose)?

Ah I think it clicked for me - if we allow "coin-joined" txs with multiple kernels to be pushed onto the network then we no longer know if any inputs/outputs have been cut-through as a result of the merged txs. So it is not just a miners that could be colluding, any party in the network could have done this.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants