/srv/irclogs.ubuntu.com/2020/03/31/#smooth-operator.txt

=== max__ is now known as Guest94867
=== max__ is now known as Guest28910
jamSo the reason some charms end up with '-k8s' in the name, is because in the current charm store, you can't have the same name for different purposes.07:11
jam(k8s vs normal iaas charms)07:11
jamThe intent with the new charm store was to leverage a field like Arch to allow you to 'juju deploy mysql' which if you are in a k8s model is a different charm than if you are in a AWS model.07:12
jamSo the goal is to get rid of -k8s, but it is a necessary wart today.07:12
jamthanks for the update Dmitrii-Sh . You really should sleep earlier. :)07:13
Dmitrii-Shjam: you're right07:14
Dmitrii-Shjam: regarding the charms, one thing I noticed is that VIPs make functional testing harder07:15
Dmitrii-Shthere is an expectation that something will be pre-allocated and is not modeled as an abstraction07:15
jamDmitrii-Sh, we talked in Juju about having a way for a charm to say "get me another IP from the provider" which it would then use for its purpose.07:16
jamIn practice, it was just a testing thing, because all the charms that have VIP end up assigning real meaning to them in the rest of the system07:16
jamso an arbitrary value wasn't good enough.07:16
jamThey *cared* that it was 10.100.20.50 because that was the firewall rule that they set in the some other system.07:17
Dmitrii-ShI agree, in all cases I've seen so far it was the case. Plus there was a domain name associated with that vip07:18
Dmitrii-Shjam: w.r.t https://github.com/canonical/operator/issues/193, instead of inheriting in ops/testing.py I could recreate the same type with a different name and __bases__ and __dict__ preserved07:20
Dmitrii-Shthat wouldn't help with mutable class attributes though07:20
Dmitrii-Sh(inheritance doesn't help either so it's not better or worse in a sense)07:21
jamDmitrii-Sh, it sounds like we're doing something bad with SavedState if we can't subclass a type that has it and have it work.07:21
jamDmitrii-Sh, that said, we'd rather not subclass and just fix the attr thing to change attrs on the instance rather than the class07:21
Dmitrii-Shjam: so far we have .on, .state (or whichever other name a dev picks) and .handle_kind as class attributes. define_event works with types that inherit from EventSetBase so define_event would have to be done for every instance of a type that inherits from EventSetBase.07:41
Dmitrii-ShDo you have any ideas on where that binding to instances would take place?07:41
jamDmitrii-Sh, I haven't worked out exactly how it would hold together. Maybe EventSetBase.07:45
jamWe could do something like CharmBase.__init__, but I don't think we want people to have to re implement that for every thing they want to create that emits events.07:46
* jam lunches08:29
Chipacagooood (belated) morning peeps!08:29
niemeyerGutten Morgen08:30
Dmitrii-ShChipaca: https://trello.com/c/mc1YggOd/9-top-10-15-charms-in-op-fmk there is a card to identify top 15 charms to be written in the operator framework. I wrote down the ones that I think would we worthwhile to consider based on the charm store stats.08:37
ChipacaDmitrii-Sh: nice08:37
ChipacaDmitrii-Sh: missing two inputs into that: what field thinks is hawt-and-upcoming, and any special requests :-)08:38
Chipacaalso, where are my glasses08:39
Dmitrii-ShThe most popular ones in the store are the OpenStack charms but I excluded most of them (except for rabbitmq, mysql and hacluster).08:39
Chipaca👍08:39
Chipacathank you08:39
Dmitrii-Shmysql is one of the charms that OpenStack eng recently rewrote in reactive so I am not sure if we need to rewrite it yet again as a first candidate08:40
Chipaca:-) no need to start there, indeed08:42
Dmitrii-ShChipaca: as for input from the field, I can go and ask them in their channel. I know that Arno was keen on having OSM k8s charms in the new framework.08:42
Dmitrii-ShI am sure they'd appreciate better landscape charms08:42
ChipacaDmitrii-Sh: ask them for a top 5 (and make it clear it's not a promise)08:43
Dmitrii-Shok08:43
jamChipaca, can you see the message above about -k8s ? I can repeat it if you cannot08:54
Chipacajam: not with this account, could you repeat?08:55
jam<jam> So the reason some charms end up with '-k8s' in the name, is because in the current charm store, you can't have the same name for different purposes.08:55
jam<jam> (k8s vs normal iaas charms)08:55
jam<jam> The intent with the new charm store was to leverage a field like Arch to allow you to 'juju deploy mysql' which if you are in a k8s model is a different charm than if you are in a AWS model.08:55
jam<jam> So the goal is to get rid of -k8s, but it is a necessary wart today.08:55
Chipacajam: yep08:56
Chipacajam: but we're not getting platform for a while yet08:56
Chipacajam: also, also08:56
Chipacajam: i understand it's often the case that the k8s charm and the non-k8s charm is frequently authored by different teams08:56
Chipacajam: in that case, -k8s in the name will continue to be a thing08:57
jamChipaca, indeed. I'm just pointing that -k8s feels like a wart, when you look at "kubectl get all" and see a bunch of foo-k8s-operator objects.08:57
Chipacaor as per gustavo's idea yesterday <foo>-k8s-operator08:57
Chipacajam: that is fair. There is nothing in the store's roadmap to fix that wart though.08:58
niemeyerMy idea was -operator, to be fair :)09:00
niemeyerChipaca: Not sure I get the relationship between that wart and the store?  How's that a wart in the store?09:01
Chipacaniemeyer: the wart can only be fixed by store-side work09:02
niemeyerThat's what I understood the statement to mean, but I don't get what that's actually referring to09:02
Chipacaniemeyer: ownership is by name, you can't have different owners for different platforms under a single name09:03
niemeyerChipaca: We do have delegation in the roadmap, and that seems like an aside.. we have support for independent channels for different architectures, and will have for different platforms as well09:04
Chipacaniemeyer: where is delegation in the roadmap? (is this what delegation is? seems a strange name for it)09:07
Chipacaniemeyer: i don't understand what you mean by it being an aside09:07
niemeyerChipaca: Delegation of authority09:08
Chipacain on-prem enterprise stores, yes09:08
Chipacathat does not sound like this09:08
niemeyerChipaca: It's an aside in the sense that when it becomes an issue we can prioritize the work.. the critical piece of that work is that we need to support orthogonal channels for orthogonal platforms, and that's part of the design right now09:09
niemeyerChipaca: It'll be an excellent problem to have when entirely different authors are fighting to maintain the same charm in different platforms09:09
Dmitrii-ShChipaca: I sent an e-mail to their list (CC-ed) and a message to their channel soliciting some feedback. Hopefully we can get some stats out of that.09:10
ChipacaDmitrii-Sh: thank you09:10
jamhttps://github.com/canonical/operator/pull/196 pulls out some of the stuff from my test_model.py patch, as the Harness changes were relevant for interface-pgsql09:16
Chipacajam: did you see i tagged you on issue #193?09:18
jamChipaca, yeah. Dmitrii-Sh and I were chatting about it earlier.09:18
Chipacaah ok09:18
jamWe can change how we create the class, but it feels like we're doing something wrong if you can subclass something that has state.09:19
jamthat said, we'd rather not subclass if we can change how the accessors are done.09:19
niemeyerHow to subclass something that does _not_ have state?  Isn't a class without state basically a function? :)09:20
jamniemeyer, if you do class MyClass: state = StoredState(); having class NewClass(MyClass): will break.09:21
jamat least, that is how I read https://github.com/canonical/operator/issues/19309:21
niemeyerShould we fix that?09:21
jamyes. my argument was we should fix that regardless, even if we don't want to create a subclass for testing.09:22
niemeyerI'm not suggesting we should encourage subclassing of everything, but it sounds like a slightly awkward hard constraint09:22
niemeyerYeah, same09:22
Chipacawill make writing components as mixins hard for example :)09:22
niemeyerIt just feels like it's an artificial constraint.. we never considered it an intended limitation09:23
niemeyerWe can discuss when and whether mixins/subclassing/etc make sense, but right now we can't09:24
niemeyerIt's probably an easy fix as well.. did you look into it jam?09:25
ChipacaStoredState is gnarly, undocumented code09:26
Chipacais the attr_name thing it's doing just doing what we do with set_name elsewhere?09:26
niemeyerIt is indeed.. it exists so that people can not worry about that sort of thing09:26
jamI haven't looked deep into it yet.09:27
ChipacaI'll see if I can't make the code a bit clearer, unless you're already looking at it09:29
jamI'm not right now. looking at interface-pgsql currently09:29
Chipaca👍09:31
Chipacaniemeyer: so the intent is for any class to be able to have a stored state, even if it doesn't inherit from Object?09:40
niemeyerChipaca: Object is not as important as it looks like for the framework.. logic generally check for what they need, not for inheritance09:41
niemeyerChipaca: Object is there for developer convenience09:42
Chipacaniemeyer: so it doesn't have to Object, but it does have to look like Object09:42
niemeyerChipaca: If that was the intention then we'd check for inheritance09:43
niemeyerChipaca: The point being that Object may "look like" things that we don't care about09:43
niemeyerChipaca: Makes sense?09:44
Chipacait's duck typing, but without being explicit about what's needed09:44
Chipacano it does not make sense :)09:45
Chipacacurrently if you just try to use StoredState like this,09:45
Chipacaclass MyC:  stored = StoredState()09:45
Chipacaand then do  c = MyC(); c.foo = 4209:45
Chipacait fails, because it tried to call register_type on a 'framework' attribute09:46
niemeyerChipaca: If class C1 depends on attribute A1 and A2, and class C2 offers A1, 2, 3, 4, 5, 6, and 7..  does class C1 depend on how class C2 looks like, or depends on A1 and A2??09:46
Chipacaniemeyer: if class C1 does not document what it expects of the things it's usually used with, and fails in obscure ways as soon as you try to use it outside of that, it might as well depend on every detail of the implementation09:46
niemeyerChipaca: If the water is dirty, do you throw the baby out too?09:47
niemeyerChipaca: Or do you document the baby and clean it up?09:47
Chipacawhy are you talking to me like this last thing isn't _EXACTLY_ why i started asking the question, dammit09:48
niemeyerChipaca: I'll wait until you relax so we can continue the conversation..09:48
* niemeyer steps out and sips some mate..09:48
facubatistaMuy buenos días a todos!11:08
facubatistaDmitrii-Sh, hola! question about gh#193, you say that "having a state attribute on a charm currently raises an exception."... that means that if you have a real charm class, with a class attribute `state = StoredState()`, it will crash?11:15
facubatistaAh, I see now that that conversation continued later11:18
* facubatista keeps reading11:18
Dmitrii-Shfacubatista: yes, pretty much11:23
facubatistaDmitrii-Sh, sounds to something we should allow to be tested, not something that should be worked around11:24
facubatistaDmitrii-Sh, otherwise we're giving the signal to developers that is a bad idiom of the charm11:24
facubatistajam, ^11:24
jamfacubatista, that seems to be the consensus. The specific issue is that testing.Harness uses a subclass, which turns out to break.11:25
jamand we don't really want to use a subclass for testing, but we also don't want to break if someone does subclass.11:25
facubatistajam, perfect, thanks11:26
Dmitrii-ShSorry for lagging - had a call with David from the OSM team.11:31
Dmitrii-ShYes this is definitely something we need to have built-in. All of my current charms use state in some way at the moment, for example.11:31
jamDmitrii-Sh, yeah, the component I'm testing uses state, but isn't a complete charm, so I hadn't run into it yet.11:35
jamThe 'charm' I'm interacting with is just CharmBase, and then I use that to fire events into my component.11:36
facubatistaniemeyer, I tried to build a simple class that uses StoredState... I end up depending on having a Framework... yes, Object is not that important per se, but this "implicit framework needed floating around" is what makes me incomfortable in some situations11:38
facubatistaNot saying we should change that, but maybe we could improve something about it11:38
facubatistafor example:11:38
facubatistaclass DBLoggerInterface(Object):11:38
facubatista    def __init__(self, charm, name):11:38
facubatista        super().__init__(charm, name)11:38
facubatistaHaving to call Object with the `charm` looks arbitrary11:39
facubatistaand that's only because "or the parent is the framework, or the parent has a framework"11:40
facubatistaso, it looks like the rule is "there must be a framework floating around for everything" (which is fine, this is the operator framework), but "it's not really floating around, everybody should have it", maybe we need to make that more explicit11:44
* facubatista is just thinking out loud, doesn't have a concrete proposal11:45
niemeyerfacubatista: Let's cover that in the standup in a bit.. first thing we need to figure out is why specifically subclassing was breaking down. This is unrelated to what is passed in or whether the object has a framework or not. If the base class worked, the subclass should work too, right?11:57
facubatistaniemeyer, absolutely, that's out the question for me; subclassing must work11:58
Dmitrii-ShThe attribute lookup (https://git.io/Jv5uv) is performed in a __dict__ of the type that is supposed to hold the StoredState attribute. When we subclass the type that holds '.state = StoredState()', the new type no longer holds '.state' attribute in its __dict__ (its parent type does).12:16
Dmitrii-Shwhat I have in the pdb output in the bug:12:17
Dmitrii-Sh(Pdb++) parent_type.__dict__.items()12:17
Dmitrii-Shdict_items([('__module__', 'ops.testing'), ('on', <ops.testing.Harness.begin.<locals>.TestEvents object at 0x7f935be611f0>), ('__doc__', None)])12:17
Dmitrii-Sh(Pdb++) parent_type.__bases__[0].__dict__.items()12:17
Dmitrii-Shdict_items([('__module__', 'src.charm'), ('state', <ops.framework.StoredState object at 0x7f935c486400>), ('HAPROXY_ENV_FILE', PosixPath('/etc/default/haproxy')), ('__init__', <function HaproxyCharm.__init__ at 0x7f935c06faf0>), ('on_install', <function HaproxyCharm.on_install at 0x7f935befc160>), ('on_start', <function HaproxyCharm.on_start at12:17
Dmitrii-Sh0x7f935befc1f0>), ('on_stop', <function HaproxyCharm.on_stop at 0x7f935befc280>), ('on_config_changed', <function HaproxyCharm.on_config_changed at 0x7f935befc310>), ('on_backends_changed', <function HaproxyCharm.on_backends_changed at 0x7f935befc3a0>), ('__doc__', None)])12:17
jamDmitrii-Sh, then could we just do getattr(obj) instead of __dict__ ?12:25
jamor if we're intentionally avoiding __get tricks, then object.__get_attribute__ ?12:26
Dmitrii-Shgoing to try that now12:28
Dmitrii-Shhmm, self.attr_name can be None12:32
Dmitrii-Shjam: the first __get__ that will be run is going to have self.attr_name == None12:37
Dmitrii-Shand then the for loop will be invoked to look up an attribute name by an attribute value12:38
Dmitrii-Shso we also need to take care of this: `parent_type.__dict__.items()`12:38
Dmitrii-Shjam: e.g. `for attr_name, attr_value in inspect.getmembers(parent_type): ...`12:44
niemeyerFolks, we don't need to *all* be working on the exact same issue12:45
jamafaict Dmitrii is the only one actively coding, I'm giving feedback on how I think it could work.12:53
Dmitrii-Shusing getmembers leads to an infinite recursion in some framework test cases so I'll need to explore more13:03
jamso for things that do getattr hacks (as we tend to do here), you can often use object.__getattribute__(member) to avoid the classes' __get function (IIRC).13:10
facubatistajam, these methods not implemented in the testing, e.g.13:12
facubatista    def storage_list(self, name):13:12
facubatista        raise NotImplementedError(self.storage_list)13:12
facubatistajam, should be implemented by whom? where self.storage_list should come from?13:12
jamfacubatista, they are part of the ModelBackend, and should be implemented in the Harness, just didn't get to the full interface13:13
jamsorry, should  be implemented in _TestModelBackend, I had NotImplementedError there for us to know what things to add.13:13
facubatistaah, for the future we should have a comment there (ideally, with name and date, like: XXX 2020-03-25 jam: need to implement this in future branches)13:14
facubatistajam, do we have an issue open or a trello card to track this missing work?13:14
jamnot that I'm aware of.13:14
facubatistajam, we should, otherwise we'll forget about it13:15
facubatistajam, would you please? thanks!13:15
* facubatista is not really sure if in English that "would you please" is like in Spanish having it implicitly the "take care of", or sounds harsh -- if it sounded bad, was not my intention13:16
jamfacubatista, no, it is a genuine gentle request.13:17
facubatistathanks13:17
jamor at least I interpreted it as such. (probably you could do it otherwise :)13:17
Dmitrii-ShCreated draft PRs for the charms I have at the moment: https://github.com/canonical/cockroachdb-operator/pull/1 https://github.com/canonical/keepalived-vrrp-operator/pull/1 https://github.com/canonical/haproxy-operator/pull/113:20
Dmitrii-Shthe repositories under canonical are empty except the license file13:20
* facubatista brb14:03
karimsyeI am rewriting the charmed-osm/mariadb-k8s charm with the operator framework and I am having some trouble providing the mysql relation and relation data.14:09
karimsyeDmitrii-Sh: I tried to implement it by following the gitlab-k8s example but couldn't get it to work.14:10
Dmitrii-Shkarimsye: I am up for a call to give you some guidance if you have time14:14
* jam eod14:14
facubatistakarimsye, can you pastebin what you currently have? thanks!14:21
karimsyeDmitrii-Sh: ok thanks. I have to run an errand, I should be back in about 15 minutes14:26
Dmitrii-Shkarimsye: ok, sgtm14:26
Dmitrii-ShI'll wait14:26
karimsyeDmitrii-Sh: https://meet.google.com/uck-qvcz-aoq14:59
Dmitrii-Shkarimsye: 2 min, need to send an e-mail15:01
karimsyeDmitrii-Sh: np, I am trying to setup my environment as well15:01
karimsyefacubatista: the git repo for my charm (the files in src are not the latest) - https://github.com/SMAK1993/charmed-osm-mariadb-k8s15:06
karimsyefacubatista: my latest src/charm.py - https://pastebin.canonical.com/p/R8sKN7FWW3/15:06
karimsyefacubatista: another file being used src/interface_mysql_provides.py - https://pastebin.canonical.com/p/ZCVP43qBcV/15:07
karimsyeinterface_mysql_provides.py is my version of https://github.com/johnsca/interface-http/blob/master/interface_http.py15:07
karimsyebut trying to implement the mysql interface/relation15:08
facubatistakarimsye, reading it15:10
karimsyethanks15:10
karimsyefacubatista: in charm.py the class MariaDbProvides at the bottom was my attempt to imitate https://github.com/tvansteenburgh/charm-kine/blob/master/lib/charm.py. The current charm.py isnt using it but when I tried to use it in a similar manner to the charm-kine example, I wasn't able to successfully relate to cs:~charmed-osm/keystone-k8s15:13
facubatistakarimsye, let me understand better the behaviour you're NOT getting... if you deploy this code, where is the problem, or when the problem arises? or you didn't get to deploy it yet?15:15
karimsyefacubatista: I deployed it and the error I got was:15:17
karimsyehttps://pastebin.canonical.com/p/TDZD7yhNMq/15:17
karimsyebasically EventBase does not inherit from Object15:18
* facubatista checks the error15:20
Dmitrii-Shfacubatista: on a call with karimsye and Narinder now15:26
facubatistakarimsye, ok, the code looks good AFAIC (by just looking it) for everything related hooking the charm and the interface together and have stuff flying around15:26
facubatistagoing in15:26
Dmitrii-Shevents are not Objects themselves so there's an issue in using shortcuts15:26
niemeyerfacubatista was going to look into this.. we should be able to make this work as well15:30
Chipacaniemeyer: do you have a moment to go over StoredState?15:32
niemeyerI do in about an hour or so15:38
Chipacaniemeyer: works for me15:38
facubatistaniemeyer, regarding the "having shortcuts in events" I never got a conclusion, got lost in the code, I can hit another round to it15:52
facubatistaniemeyer, regarding current problem, was something else, really, funnily we could locate it after applying a patch from this branch https://github.com/canonical/operator/pull/195/files15:53
Chipacafacubatista: ping?16:06
facubatistaChipaca, pong16:11
Chipacafacubatista: meet plz16:11
Chipacahmm16:36
Chipacaniemeyer: not sure if you saw my ping, or if you replied16:36
facubatistaChipaca, nothing from you since the "meet plz" to me and your "hmm"16:36
Chipacasomething seriously wrong with my thunderbolt ethernet thing16:36
Chipacafacubatista: ack, thanks16:36
Chipacaniemeyer: ping :)16:36
Chipacanice tracebacks in dmesg16:37
niemeyerChipaca: Just grabbing a cuppa and will be with you16:42
Chipacaaugh, machine is now all weird16:43
Chipacarebooting, bbiab16:43
niemeyer> niemeyer: not sure if you saw my ping, or if you replied16:46
niemeyerChipaca: There was no ping here, FWIW16:46
Chipacaniemeyer: yep, facu told me so (so then i pung)16:46
niemeyerChipaca: I'm ready when you are16:47
Chipacanow back, without the thunderbolt ethernet16:47
Chipacaniemeyer: standup meet work for you?16:47
niemeyerYeah16:47
Chipacaok16:47
Dmitrii-Shfacubatista: alright, I am done with the call. For the purposes of this charm, storing their names is enough.17:01
facubatistaDmitrii-Sh, wonderful!17:02
facubatistaDmitrii-Sh, I wonder about the other code, though, that was storing the apps17:02
Dmitrii-Shthe idea there is that .serve needs to be called once per relation-id17:03
Dmitrii-ShCory's original code did it by storing app names17:03
Dmitrii-Shso if you've seen an app already, you don't need to write relation data to your own unit's data bag multiple times17:04
Dmitrii-Shit is an idempotent operation which doesn't generate new -changed events for remote units if there are no data changes but this code seems to be avoiding extra relation-set calls17:04
karimsyeDmitrii-Sh, facubatista thanks again!17:18
facubatistakarimsye, anytime!17:18
Chipacaniemeyer: I need to go make dinner, but thought I'd point you at this before going17:29
Chipacahttps://github.com/canonical/operator/compare/master...chipaca:heritage#diff-6753f94ec7178cc8798db3ae94d35100R125517:29
Chipacaneed to figure out what's going on, but if I uncomment that line, the test suite hangs17:30
Chipacathat's generally considered a bad thing17:30
niemeyerChipaca: Test is creating a framework at the end without closing it18:10
niemeyerChipaca: It must be hanging while opening the next framework, waiting for the former one to go away18:10
Chipacaniemeyer: THANK YOU, that was it19:30
Chipacai think we should split the stored state out into its own file; it deserves its own book in the docs :)20:07
ChipacaEOD here20:10
Chipacasupermarket has changed its timetable, so that was a waste of time. I'll have to go tomorrow morning early.20:44
facubatistajam, there I commented on #19620:49
facubatistajam, beyond that, we need to talk about _TestingModelBackend... some of its private structures (_relation_ids_map, _relation_names, _relation_list_map, _relation_data) are being used/changed from *outside* the class20:51
facubatistajam, that could potentially lead to incongruences from those structures (like, a relation id being in _relation_ids_map but not in _relation_data) that would lead to super subtle bugs in the future20:51
facubatistajam, it's kind of fine to have several structures to access the same "data corpus", but we need to ensure consistency20:52
facubatistajam, anyway, let's talk on Thursday20:52
facubatista(I'm off tomorrow)20:52
facubatistaso! see you all on Thursday!20:53
* facubatista eods20:53

Generated by irclog2html.py 2.7 by Marius Gedminas - find it at mg.pov.lt!