-
Notifications
You must be signed in to change notification settings - Fork 212
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
Enable contract listeners with multiple filters #1418
Conversation
Codecov ReportAll modified and coverable lines are covered by tests ✅
Additional details and impacted files@@ Coverage Diff @@
## main #1418 +/- ##
==========================================
Coverage 100.00% 100.00%
==========================================
Files 324 325 +1
Lines 23706 24019 +313
==========================================
+ Hits 23706 24019 +313 ☔ View full report in Codecov by Sentry. |
@nguyer (fyi @EnriqueL8) - there is a really good reason why we need this (at least prior to this PR). That's because calculating the signature of a listener is not an easy thing to do - and FireFly doesn't expose any way for code outside of FireFly to do it. So if you think about the common startup reconcile in most applications, where it does a query-then-create on the listeners it needs, you need to be able to match the things you get back, to the things you need. That means you need to be able to create the signature for the things you need (not just the things you create). Currently, we avoid this by having it so that if you try and recreate the new thing with the same signature, it prevents you with a 409. In fact that's important enough that we found and fixed a bug in it recently in #1433. Now I can imaging this PR might provide a proposal for a fundamentally new way to do this, by pushing the complex reconcile logic inside of listener. But we do need a clear migration path, and to know what the impact would be on apps relying on the old behavior - because without any thought they would duplicate a new listener on every startup. |
@peterbroadhurst I'm trying to pick this back up but really not sure what direction to take it. I think part of my struggle is that I myself haven't run into the use case that is driving the need for this functionality, so I'm sort of designing a solution to an unknown problem. I could add an updated version of the constraint we used to have (and maybe do some heavy DB schema changes to support that) but what would the constraint be? Topic/Location/[SOMETHING]. Is that last element of the constraint the combined signature of each filter in the listener? Do we need to prevent someone from creating the same exact Topic/Location/[Combination of Signatures] or do we want to prevent the same Topic/Location/[Single event signature] from being used in multiple listeners even if the set of filters in each of those listeners is different. Does that question make sense? I feel like this PR probably needs another round of design discussion before it's ready for more coding. |
Signed-off-by: Nicko Guyer <[email protected]>
Signed-off-by: Nicko Guyer <[email protected]>
Signed-off-by: Nicko Guyer <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
New approach is that we have introduced a new API at PR needs some conflict cleaning and test coverage but test in a FF CLI setup and looking good! |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hey @EnriqueL8 - lots of mini comments here, but it all rolls up to one big question about the relationship between:
len(filters) == 1
sosignature
is likeSomeEvent(1,2,3)
len(filters) > 1
so:- Each
filter
has asignature
like likeSomeEventA(1,2,3)
etc. is queryable in thefilter[0]
JSON - The
signature
of the listener ishash(filters[*].signature)
- Each
Then the implication of this is that signature + topic
is still a uniqueness criteria on the listener (assuming it is at 1.3.0)
db/migrations/postgres/000117_update_contractlisteners_add_filters_column.up.sql
Outdated
Show resolved
Hide resolved
|
||
filters := make([]*filter, 0) | ||
// This keeps the existing behaviour where a contract listener could only listen to one event | ||
if listener.Event != nil { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Feels odd to me that it's the blockchain/ethereum
layer responsible for this array-ification.
Why isn't this done for us in a consistent deprecated field handler, in the FF/API layer, before calling this always with an array.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If then a blockchain connector supports only len(filters) == 1
, it can return an error
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah makes sense - I think this was done before for reload of contract listeners on restart, but how not standardise that on load of listeners from DB instead of here!
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Need to address this now
internal/contracts/manager.go
Outdated
@@ -835,6 +901,50 @@ func (cm *contractManager) AddContractListener(ctx context.Context, listener *co | |||
listener.Options.FirstEvent = cm.getDefaultContractListenerOptions().FirstEvent | |||
} | |||
|
|||
// Normalize Event/EventPath + Location to list of Listeners | |||
if len(listener.Filters) == 0 { | |||
listener.Filters = append(listener.Filters, &core.ListenerFilterInput{ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is what I was expecting to see as the migration from the deprecated fields.
The fact this is in the manager
level, makes me confused why any other code apart from this one bit needs to worry about the fact that "on the API boundary we have a special spelling for handling for len(filters) == 1"
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Have added this at the manager level and also still provided the deprecated fields in the API
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Address this in the blockchain plugin
For the second case, we need to include the location if not it will not be unique and will cause problems |
@peterbroadhurst Thanks for the comments - will amend the PR with changed. I think the challenge here is the
|
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
@peterbroadhurst From the review comments and the chances I made, came up with three other worries:
|
Signed-off-by: Enrique Lacal <[email protected]>
@EnriqueL8 tezosconnect doesn't support event streaming yet, so don't worry about compatibility with this connector |
@denisandreenko thanks for the input - so a follow up PR might be needed to provide the user with a better error on event streaming because it seems the code does go ahead and try and create a subscription in the Tezos connector |
Looks like Docker has been update in the Github Action and the packaged docker-compose version no longer support |
The GH action is using V1 for some reason |
This should do the trick hyperledger/firefly-cli#307 - I'll have to update the action to pull in a new version |
This comment you made is highlighted, but there's no commentary on where you got to with it @EnriqueL8 . |
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
internal/contracts/manager.go
Outdated
// comparing to channel + chaincode | ||
if strings.Contains(duplicateMapChecker[filter.Signature], location) || strings.Contains(location, duplicateMapChecker[filter.Signature]) { | ||
return i18n.NewError(ctx, coremsgs.MsgDuplicateContractListenerFilterLocation) | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I do wonder if this should be moved to the plugin specific code... or the plugin should generate a signature including the location
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What's your proposal here @EnriqueL8? Are you suggesting you make a change before this goes in?
I think your point that requiring location to be part of the signature is valid - but you'd be best positioned to comment on the cost vs. benefit in the code I think.
for _, listener := range existing { | ||
// We have extended the event signature to add more information | ||
// So we compare the start with is not guaranteed to be the same | ||
// but it's the best comparison | ||
if strings.HasPrefix(signature, listener.Signature) { | ||
return i18n.NewError(ctx, coremsgs.MsgContractListenerExists) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Again maybe this comparing of signatures should be plugin specific
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think comparing signatures should be specific, but I don't understand the HasPrefix
here.
Surely the signatures are the same, or they are not.
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Next round of detail comments/questions up for your @EnriqueL8 - these are all around the "how" I think at this point, so that's great.
Pausing here as I do think there's some significant questions here for you to think about before I go further.
db/migrations/postgres/000117_update_contractlisteners_add_filters_column.up.sql
Show resolved
Hide resolved
@peterbroadhurst Just working through the changes came across |
I think it's quite easy to sort it by checking if the whole signature contains the current event signature and location! |
Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
Great code coverage is working, fixing Weird I'm getting 100% locally for coverage |
Signed-off-by: Enrique Lacal <[email protected]>
Co-authored-by: Peter Broadhurst <[email protected]> Signed-off-by: Enrique Lacal <[email protected]>
Signed-off-by: Enrique Lacal <[email protected]>
The Postgres migration seem to be failing, looking into it.. It's a shame the GitHub actions do no give you logs of the docker containers, something to add to my list |
Replicate locally Changing it to just VARCHAR which according to the Postgres documentation is unbounded https://www.postgresql.org/docs/current/datatype-character.html |
Signed-off-by: Enrique Lacal <[email protected]>
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
✨ 🚀 🌑
This PR adds the ability to listen to multiple types of events on the same contract listener, by adding an array of listeners, rather than a single event signature/location per listener. The old way of creating a listener is still accepted by the API, but it will always be returned in the filters array now. This is a migration concern that needs to be documented.
Open questions
One thing I'm not sure about here, is that this PR as-is removes the uniqueness constraint on listeners by topic/location/signature. It now allows multiples. I'm not sure if this is a problem or not. I can add that constraint back, but it would likely require some more sophisticated DB changes. Which brings me to the next point...
Right now all the filters for a contract listener get serialized to JSON and stored in a single column. I lated realized this means we lose the ability to query/filter (no pun intended) by signature, location, etc. which we used to do, in order to check for duplicates. I'm not sure if this is required or not, but wanted to call it out.
Example
Create contract listener request
Create contract listener response