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

Durable users - WIP / for discussion #78

Draft
wants to merge 8 commits into
base: master
Choose a base branch
from
Draft

Durable users - WIP / for discussion #78

wants to merge 8 commits into from

Conversation

devth
Copy link
Member

@devth devth commented Mar 28, 2019

Opening this up as a place to discuss how we want to model users.

Goals of this discussion

  1. Provide a consistent and durable model of users to build on top of
    • Karma would gain a foreign key to users
    • GraphQL resources like history and karma would be able to link to the user entity
    • We would be able to reliably build user prefs on top of it
  2. Figure out how and if Yetibot should support multiple adapters. Users are the primary entity that we need to consider, but all db state should be evaluated.

Multi adapter

DB entities to consider:

  • alias
  • channel
  • cron
  • history
  • karma
  • observe
  • status
  • user

Two extremes:

  1. be truly multi-adapter, unifying users (i.e. if I'm logged in on both users the two user entities should be somehow linked and represent a single identity) and all state across all connected adapters
  2. vhost model: as if each adapter was its own Yetibot instance with all state constrained

Current state

Currently Yetibot supports multiple adapters but it's somewhere between the two extremes.

Considerations

Feature Pro Share Pro Isolate
alias If an operator creates an alias on one adapter they'd probably expect it to be available on another adapter Sharing aliases between adapters could possibly leak team specific information
channel Channel configuration is already scoped to individual channels by default
cron Cron configuration is already scoped to individual channels by default
history History is scoped to the channel and adapter it originated from. Whether we expose it beyond that is a choice we can make/change later.
karma It might be weird to see users from another adapter (that you may have never heard of) in the Karma list, especially if you see yourself multiple times because users are not unified. If you are a user in multiple adapters, you might expect to see a unified karma list
observe Same logic as alias - you may expect to see observers working in all adapters that Yetibot is listening in. Observers have the option of only firing for specific channel patterns, but that means it could match multiple channels across multiple adapters Shared observers could leak team information
status Currently stores the adapter and channel that it was created in. Whether we expose can be decided and changed later since we have all provenance the data.
user Wouldn't want to see a single identity represented by multiple users It's hard to unite multiple representations of a single identity

Current use cases of multiple adapters

  1. A single public Yetibot currently listens on both freenode and Slack. This is the shared case, where any state is expected to be exposed in all adapters.
  2. We run a single Yetibot internally at eBay with multiple configured Slack Adapters - only because Slack does not yet support the concept of multi-workspace bots within a single org. This is shared and all state is expected to be shared across all adapters (including users, since there is technically a single org user). However: we confirmed that the underlying user ID is actually not the same. This appears to be a weird hack that Slack used when they launched Organizations (aka Enterprise Grid). I'm guessing in the future:
    • these user IDs might be unified
    • and they will enable org-level bots that can listen across all/any workspaces, therefore mitigating this entier problem

Concluding thoughts

  1. If we capture provenance (which adapter and channel a record originated from) we can alter decisions like what state is exposed where down the road
  2. If we can stitch users together we get much closer to an ideal shared-state approach (i.e. anti-vhost), especially for things like karma where you would never want to see a single identity represented by multiple users, but this could be hard. If we model it in a way that allows tieing an identity to multiple representations, we could require the user to manually specify all their IDs until we could automate it (e.g. my IRC id is ~devth and my Slack ID is @U123123 - maybe this is akin to connecting an identity on a website to disparate oauth identity providers like twitter, fb, github, etc)
  3. What if we do nothing except capture the provenance?
    • We still have the capability to run multiple adapters
    • Some entities are leaked by default, like aliases and observers
    • After karma gains provenance we could opt to constrain leaderboards to adapter, or allow it to span all adapters
  4. Regarding user stitching: we could unify users if their username is identical, thereby avoiding the problem with single-org multi-workspace users having different ids in each workspace. This could even be configurable (:attempt-to-stitch? true).
  5. Why not just give up the multiple adapter functionality and simplify everything?
    • It's currently a workaround for Slack's weird enterprise grid limitations where otherwise we'd have to run/operate/maintain multiple Yetibots
    • In the case of constrained resources (public Yetibot runs on a 1gb DO Droplet), we couldn't run multiple instances even if we wanted to since it requires multiple jvm runtimes, which are not cheap

@devth devth requested review from cvic, jcorrado and kaffein March 28, 2019 19:17
@devth devth force-pushed the durable-users branch 2 times, most recently from 6992671 to ffd1f06 Compare March 29, 2019 13:37
@devth
Copy link
Member Author

devth commented Mar 29, 2019

From @jcorrado via Slack:

I'm still of the "all of nothing, do it cleanly or don't do it at all" mindset (edited)
but, I understand practical middle-ground (in theory)
for me the simple thought experiment goes: were we to simplify to one adapter, what would that simplified, clarified architecture open up?
and is that worth more than the what would be lost

@devth
Copy link
Member Author

devth commented Mar 30, 2019

were we to simplify to one adapter, what would that simplified, clarified architecture open up?

Corollary: What if we just design everything as if there were only one adapter (while still capturing provenance)? Rationale:

  1. I'm probably only person running multi adapter, so every operator of Yetibot aside from myself would still have clean single-adapter state and
  2. we wouldn't have to invest any time into stitching user accounts and making everything truly multi adapter

This is essentially what I've been doing all along and it hasn't hindered me. The remaining TODOs for Users would still be:

  1. add provenance to all state (good idea, regardless)
  2. model users, including provenance

I think of it as a stepping stone to the ideal "all or nothing" state.

This means both sets of users show up in karma. Observers, Status, and Aliases are global. I think all of these are fine. I've always tried to let concrete use cases drive the design of Yetibot, and I don't think I have a strong enough multi adapter case (Yetibot public runs on Slack and IRC but no one uses it on IRC currently) to help drive its design while at the same time having a practical need for multi adapter.

In the future if/when YB supports Gitter or other platforms it will likely listen on all, and I think we want most state to be global in that case. Same with Clojure community chats: if/when we get Yetibot into Clojurians, #clojure IRC, clojure/general on Gitter, etc. When that happens I think I'll either have motivation to go all in or the clarity and confidence to make the call go single adapter.

@jcorrado
Copy link
Contributor

Actively mulling this over...

@devth
Copy link
Member Author

devth commented Apr 1, 2019

tldr: agree with all or nothing but want to defer the decision / work.

@devth
Copy link
Member Author

devth commented Apr 9, 2019

We discussed this today starting at https://yetibot.slack.com/archives/C66AJ2EFL/p1554821184926700

Summary

  1. Our goal is "true" multi-adapter
  2. We seem to be in agreement on delivering fully-baked, user-proof features. Quality UI.
  3. Even with Consider command output a as a monad preserving all metadata #2 in mind, we're practical about how we build, and are willing to cut some internal corners to deliver those features, when there's a compelling opportunity. sic: Clean features but fine to 'fix up' internals later
  4. Clojurians Slack is something we'd like to service, ideally soon to coincide with Clojure/North. We're OK levering its curtailed requirements ("Just Slack for now"), so long as features are clean. We can fix up later. This Consider command output a as a monad preserving all metadata #2 and make history room specific #3 in action.

Short term

  1. Require Slack-encoded users in the karma command. This lets us get around the fact that we don't have durable users modeled yet.
  2. Disable karma in IRC unless we can easily support it (@jcorrado will investigate)
  3. Consider removing or disabling users command until it's properly modeled

User modeling

  1. create or update durable users when the bot observes their existence: startup, channel join/leave events, polling if necessary, or lazy create-on-demand e.g. when a user runs a command or mentions another user in a command (like karma)
  2. check existence of user in the db when trying to attribute karma, and handle adapter-specific user encodings.

Mid term

  1. add provenance (chat adapter and channel) to all db state to give us flexibility on partitioning or uniting things later on
  2. begin storing users "as seen by the bot" in database for both irc and slack (might require some discussion/planning on its own)
  3. don't worry about leaking state across adapters for now
  4. consider in the back of our minds that someday we might have the notion of a single identity spread across multiple adapters (possibly an identities table that joins 1 or more rows from users)

In terms of simplifying adapters, maybe the adapter just needs to provide a function from uid to fully resolved user.

@jcorrado jcorrado self-assigned this May 21, 2019
@devth
Copy link
Member Author

devth commented May 21, 2019

Related: yetibot/yetibot#179

@@ -22,7 +22,8 @@
;; evaluation of piped expressions (i.e. %s instead of $s)
expr (binding [*subst-prefix* method-like-replacement-prefix]
(str command/config-prefix (pseudo-format-n cmd args)))
results (record-and-run-raw expr user yetibot-user
resolved-users nil ;; TODO
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure aliases would ever mention users. Either way, we can defer this to a later PR if we want.

@@ -104,7 +104,8 @@
;; piping behavior
rendered-cmd))
[{:keys [error? timeout? result] :as expr-result}]
(record-and-run-raw expr user yetibot-user)]
resolved-users nil ;; TODO
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can defer.

@@ -234,6 +234,9 @@
yetibot? (= yetibot-uid (:user event))
user-model (assoc (users/get-user cs (:user event))
:yetibot? yetibot?)
;; TODO - call (a/resolve-users event) to ensure users are resolved,
;; then pass the resulting map along to `handle-raw` as
;; `:resolved-users` below in addition to `:body`.
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a potential starting place.

  1. Add a/resolve-users function
  2. call it via (a/resolve-users adapter event)
  3. in a/resolve-users log the event and start looking at how we can pull users out of it and turn them into canonical persistent records

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

Successfully merging this pull request may close these issues.

2 participants