-
Notifications
You must be signed in to change notification settings - Fork 2
Home
class WithAuthors:
def __init__(self):
self.authors = Collection(RedisSortedSet('{}.authors'.format(self.ids.key), self.app.r.r),
app=self.app)
def on_add(self, item):
"""Subclass API."""
self.app.r.zadd(self.authors.ids.key, time.time(), item.author.id)
def on_remove(self, item):
"""Subclass API."""
if item.author.id not in set(item.author.id for item in self):
self.app.zrem(self.authors.ids.key, item.author.id)
def json(self, *, slc):
return {'authors': self.authors.json(slc=slc)}
class Comments(Collection, WithAuthors):
def __init__(self):
super().__init__()
WithAuthors.__init__(self)
def comment(self):
# ...
self.on_add(comment)
def json(self):
return {
**super().json(),
**WithAuthors.json(self, slc=slice(0:2))
}
# Maybe trashed count is not so relevant, but nice example of additional collection meta data
class WithTrashable:
def __init__(self):
self.__key = '{}.meta'.format(self.ids.key)
@property
def trashed_count(self):
return int(self.app.hget(self.__key, 'trashed_count'))
def on_trash(self, item):
self.app.r.hincrby(self.__key, 'trashed_count', 1)
def on_restore(self, item):
self.app.r.hincrby(self.__key, 'trashed_count', -1)
def json(self):
return {'trashed_count', self.trashed_count}
We step-by-step introduce type annotations for micro on a provisional bases to see how it works out in a dynamic language like Python.
Some stumbling blocks:
- Pylint does not understand @overload
- Generic class with default argument:
- Problem: User can annotate class with e.g.
Foo[A]
while default implementation usesB
- Default value for TypeVar is not implemented
- Could be solved (effortfully) with overloading
__new__
and two different subclasses
- Problem: User can annotate class with e.g.
Annotate micro as typed: http://mypy.readthedocs.io/en/latest/installed_packages.html
Cloned content of <use>
is placed and layed out
in the current container with the given width, height and preserveAspectRatio of the referenced
symbol. If width and height are set to 100%, the cloned content scales to the container (centered to
keep the aspect ratio). The container size is not influenced by the cloned content and must be
explicitly set, either for each icon (e.g. with .fa-w-X
) or for all icons to the same size.
<svg class="svg-inline--fa"><use href="{{ icons_url }}#angle-down"></use></svg>
Type hints may further improve code quality, but we should first investigate how they interact with the dynamic nature of Python. Experimental branch: https://github.com/noyainrain/micro/tree/type-hints
If the button is triggered, it will suspend (disable), then after the promise resume (enable). If the button disabled property is modified inbetween (e.g. by the promise itself), we lose this change.
// Suspend:
this.suspended = true;
this.disabledResume = false;
// Resume:
this.suspended = false;
this.disabled = this.disabledResume;
class Button {
set disabled(value) {
if (this.suspended) {
this.disabledResume = value;
} else {
super.disabled = value;
}
}
}
At the moment transforms are special functions that receive ctx as first argument. We could rather make ctx available as special variable, which can be passed to transforms that really need it.
let stack = [null].concat(data, micro.bind.transforms);
// ...
stack[0] = {ctx};
// data-content="list ctx posts 'post'"
Object.assign(this.eventTemplates, {
"purr": ".foo-purr-event-template"
});
this.eventData = {makeURL: foo.makeURL};
let elem = document.importNode(ui.querySelector(ui.eventTemplates[event.type]).content, true);
micro.bind.bind(elem, Object.assign({event}, this.eventData));
li.appendChild(elem);
- MVP / Beta 1:
- Target: App is usable in a minimal form. User mistakes are correctable.
- Audience: Groups the developers are themselves part of
- Feedback channel: Direct
- Beta 2:
- Target: App is usable on a daily basis. Core functionality is implemented. Purpose is clearly apparent to first time users. Feedback from Beta 1 is incorporated.
- Audience: Social networks of the developers
- Feedback channel: Social networks / App
- Beta 3:
- Target: App monitoring is available. Feedback from Beta 2 is incorporated.
- Audience: Public, announced on various platforms, e.g.:
- Feedback channel: App / Platforms
- Stable:
- Target: Features expected from a self-contained app are implemented. Feedback from Beta 3 is incorporated.
unwatch(prop, onUpdate) {
let i = (watchers[prop] || []).findIndex(f => f === onUpdate);
if (i === -1) {
throw new Error();
}
watchers.splice(i, 1);
}
Sophisticated user listing:
<p data-title="joinText list.authors ', ' map 'name'">
<span class="fa fa-pencil"></span>
<span data-content="join list.authors 'user' ', ' 3">
<template><micro-user></micro-user></template>
<template data-name="overflow">
+ <span data-content="sub list.authors.length 3"></span>
</template>
</span>
</p>
Format:
<span data-content="format 'Hello {user}!' 'user' user.name">
<span data-content="format 'Hello {user}!'">
<template name="user"><micro-user data-user="user"></micro-user></template>
</span>
Translate (calls format):
<span data-content="translate 'hello' 'user' user.name">
Advanced translation example:
<p>
<span data-content="translateEditedBy list.authors">
<template data-name="authors">
<span data-content="join list.authors 'user' ', ' 3">
<template><micro-user></micro-user></template>
</span>
</template>
</span>
</p>
<script>
this.data.translateEditedBy = (ctx, authors) => {
let n = authors.length;
return micro.bind.translate(ctx, n > 3 ? "edited-by-overflow" : "edited-by", "n", n);
}
{
"edited-by": "edited by {authors}",
"edited-by-overflow": ["edited by {authors} and {n} other", "edited by {authors} and {n} others"]
}
</script>
- Python / JavaScript API:
- Breaking changes okay
- User decides update time
- Sometimes non-breaking changes are not possible or only with disproportionate effort:
- (JavaScript) Convert
<custom-element>
from/to<div is="custom-element">
(can only be registered once)
- (JavaScript) Convert
- Web API:
- Breaking changes only after deprecation period
- Service provider decides update time, user needs time to make adjustments
Idea for non-breaking update path for function signatures:
def foo(a, b):
# ...
# ->
def foo(a, x, b=None, __version__=1):
if __version__ == 1:
b, x = x, 'default'
else:
if b is None:
raise Error()
# ...
# ->
def foo(a, x, b, __version__=2):
# ...
Assume that micro introduces a new announcement feature with the following API:
- type
Announcement
- Attribute
Application.announcements
- DB key
announcements
- endpoint
/api/announcements
- page
/announcements
- ...
If an existing micro application already makes use of any of those names, this could lead to breaking behavior (e.g. micro code modifies Application.announcements
or posts to /api/announcements
and they have different, unexpected semantics).
How can we solve this? New features that involve the extendable API could be major releases. New features could be introduced behind a feature toggle with a minor release and enabled by default with a major release. DB keys could be prefixed with micro
.