mirror of
https://github.com/correl/correl.github.io.git
synced 2025-01-01 11:03:18 +00:00
5193 lines
193 KiB
Org Mode
5193 lines
193 KiB
Org Mode
#+STARTUP: indent overview inlineimages hideblocks logdone
|
||
#+HUGO_BASE_DIR: .
|
||
#+HUGO_SECTION: blog
|
||
#+OPTIONS: toc:nil num:nil todo:nil d:(not "HIDDEN") tags:nil
|
||
#+PROPERTY: header-args :cache yes :eval never-export :output-dir static/ox-hugo/
|
||
#+COLUMNS: %TODO %50ITEM %CLOSED %EXPORT_FILE_NAME %CATEGORY %TAGS
|
||
#+LINK: relref file:{{< relref "%s.md" >}}
|
||
|
||
* DONE Potatoes and Portal Guns :gaming:
|
||
CLOSED: [2011-04-26 Tue]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: potatoes-and-portal-guns
|
||
:END:
|
||
|
||
#+begin_export markdown
|
||
[<img src="/images/portal_2_logo-150x150.jpg" alt="Portal 2 Logo" title="Portal 2 Logo" width="150" height="150" style="float: right" />](/images/portal_2_logo.jpg) Got my hands on Portal 2 and finished a run through the single player campaign. Was a *lot* of fun, the characters were bursting with humor and personality. Just like the first game, it was hard to stop playing. *Unlike* the first game, it's got some length, so I stayed up late a couple nights with my eyes glued to the television. I already want to play through it again to find any little things I my tired eyes may have missed.
|
||
|
||
I'm itching to give co-op a try, so if you happen to have it on xbox or care to drop by, let me know.
|
||
|
||
**Update:** Played some co-op with Jen, had fun navigating puzzles together :)
|
||
#+end_export
|
||
* DONE Meh.php :programming:
|
||
CLOSED: [2011-04-27 Wed 00:00]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: meh-php
|
||
:EXPORT_HUGO_SLUG: meh-php
|
||
:END:
|
||
#+begin_export markdown
|
||
```php
|
||
<?php
|
||
if (!defined('meh')) define('meh', null);
|
||
|
||
class Meh {
|
||
public function __set($name, $value) {
|
||
}
|
||
public function __get($name) {
|
||
return meh;
|
||
}
|
||
public function __isset($name) {
|
||
return true || false;
|
||
}
|
||
public function __unset($name) {
|
||
}
|
||
public function __call($name, $arguments) {
|
||
return meh;
|
||
}
|
||
public function __callStatic($name, $arguments) {
|
||
return meh;
|
||
}
|
||
}
|
||
|
||
$bwuh = new Meh();
|
||
$bwuh->give_a_shit();
|
||
echo $bwuh->concerns;
|
||
|
||
class SuperDuperBillingProcessor extends Meh {}
|
||
|
||
$p = new SuperDuperBillingProcessor();
|
||
$p->calculateEverything();
|
||
$p->profit();
|
||
```
|
||
#+end_export
|
||
|
||
|
||
* DONE Transmission, RSS, and XBMC :programming:python:
|
||
CLOSED: [2011-04-27 Wed 00:01]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: transmission-rss-and-xbmc
|
||
:EXPORT_HUGO_SLUG: transmission-rss-and-xbmc
|
||
:END:
|
||
|
||
#+begin_export markdown
|
||
I'm a huge fan of [XBMC](http://www.xbmc.org/). My pc (currently running Ubuntu 10.04) has taken root in my
|
||
living room, piping all my movies and tv shows straight to my HDTV.
|
||
|
||
While my pc is set up as a DVR using [MythTV](http://www.mythtv.org) to record shows off my FIOS box, it tends to be a little unreliable, which can suck when it's time to catch up on Daily Show and Colbert episodes.
|
||
I've had [Transmission](http://www.transmissionbt.com/) set up for a while for all my torrenting needs, and
|
||
I've even written an [XBMC script to manage torrents](https://github.com/correl/Transmission-XBMC), so I got to looking for
|
||
tools to track tv show torrent rss feeds.
|
||
|
||
<!--more-->
|
||
|
||
My first stop was [TED](http://ted.nu/). TED worked well enough, but would occasionally hang.
|
||
Since it's a GUI java app running in the taskbar, it would require me to dig
|
||
out my mouse and break out of full screen XBMC to fiddle with it. I eventually
|
||
got tired of dealing with TED and went back to prodding Myth.
|
||
|
||
Recently I've been itching to reliably watch my shows again, so I checked around
|
||
for a simple command-line utility to track rss feeds and download torrents.
|
||
Finding none, I loaded up vim and threw together a python script to handle it
|
||
all for me.
|
||
|
||
I also have another, simple script from when I was using TED (or just manually
|
||
downloading shows) which looks at completed torrents, compares their names with
|
||
the folders in my TV directory, and moves the shows into them for XBMC to see.
|
||
|
||
A couple cron jobs and a few rss feeds later, and I've got all my shows
|
||
automatically delivered straight to XBMC for my lazy evening viewing pleasure.
|
||
|
||
### trss.py
|
||
[Download](https://github.com/correl/trss/raw/master/trss.py)
|
||
|
||
```
|
||
Usage:
|
||
trss.py add <rss-url> [<recent-items>]
|
||
Adds an RSS feed to follow
|
||
rss-url: Full URL to the RSS feed
|
||
recent-items: (Optional) number of recent items to queue
|
||
for downloading
|
||
trss.py remove <index>
|
||
Remove an RSS feed
|
||
index: Numeric index of the feed to remove as
|
||
reported by the list command
|
||
trss.py list
|
||
Displays a list of followed feeds
|
||
|
||
trss.py download
|
||
Fetch all feeds and download new items
|
||
|
||
trss.py set [<setting> [<value>]]
|
||
Set or view configuration settings
|
||
Call without any arguments to list all settings and their values
|
||
Call with a setting and no value to see the current value for that setting
|
||
|
||
Currently, the only used setting is 'download_dir', which allows you to set
|
||
a directory to store all retrieved torrents, such as a directory your
|
||
torrent application watches for new downloads. If 'download_dir' is not set,
|
||
the current directory will be used.
|
||
```
|
||
|
||
### transmission-tv.py
|
||
```python
|
||
#!/usr/bin/python
|
||
import os
|
||
import re
|
||
|
||
import transmissionrpc
|
||
|
||
TV_PATH = '/media/Gaia/Video/TV/'
|
||
|
||
class TVShowCollection:
|
||
def __init__(self, path):
|
||
self.path = path
|
||
self.shows = os.listdir(path)
|
||
self.patterns = [[s.lower().replace(' ', '.?'), s] for s in sorted(self.shows, key=len, reverse=True)]
|
||
def match(self, filename):
|
||
for pattern, show in self.patterns:
|
||
if re.findall(pattern, filename.lower()):
|
||
return show
|
||
return None
|
||
|
||
def move(self, ids, location):
|
||
"""Move torrent data to the new location."""
|
||
self._rpc_version_warning(6)
|
||
args = {'location': location, 'move': True}
|
||
self._request('torrent-set-location', args, ids, True)
|
||
|
||
if float(transmissionrpc.__version__) < 0.4:
|
||
# The move function is not present in versions 0.3 and older
|
||
transmissionrpc.Client.move = move
|
||
|
||
collection = TVShowCollection(TV_PATH)
|
||
client = transmissionrpc.Client()
|
||
|
||
torrents = client.info()
|
||
for i, torrent in torrents.iteritems():
|
||
status = torrent.status
|
||
if status not in ['seeding', 'stopped']:
|
||
continue
|
||
show = collection.match(torrent.name)
|
||
if show is None:
|
||
continue
|
||
path = '{0}{1}/'.format(TV_PATH, show)
|
||
if torrent.downloadDir.startswith(path):
|
||
continue
|
||
print 'Found {0} torrent \'{1}\' in show \'{2}\', moving...'.format(status, torrent.name, show)
|
||
result = client.move(i, path)
|
||
if status == 'seeding':
|
||
print 'Re-starting torrent to continue seeding'
|
||
client.start(i)
|
||
```
|
||
#+end_export
|
||
* DONE Learning Functional Programming, Part One :programming:python:
|
||
CLOSED: [2012-04-09 Mon]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: learning-functional-programming-part-one
|
||
:EXPORT_HUGO_SLUG: learning-functional-programming-part-one
|
||
:END:
|
||
#+begin_export markdown
|
||
## Part One: Lambdas? In my Python?
|
||
|
||
Over the past few months, I've decided to take a stab at learning some
|
||
functional programming. I'd been doing python for a few years (and
|
||
completely falling in love with it), and so I'd been exposed to a few
|
||
functional concepts it offers - primarily higher-order functions and list
|
||
comprehensions, both of which allow for very clear, concise and powerful code.
|
||
Since that's where I started my journey, that's where my post will begin as
|
||
well.
|
||
|
||
<!--more-->
|
||
|
||
### Functions are objects, too
|
||
|
||
Having graduated to python from PHP and C/C++, perhaps the biggest new thing to
|
||
wrap my head around (besides readable code, whitespace-as-syntax,
|
||
[programming being fun again](http://xkcd.com/353/), and all that), is that in
|
||
python, functions (and classes!) are objects, just like anything else. They
|
||
can still be defined in the usual way, but they can also be assigned, passed
|
||
as arguments, even modified and replaced like any other value or object in your
|
||
program.
|
||
|
||
```python
|
||
def do_a():
|
||
print "Doing something"
|
||
|
||
do_b = do_a
|
||
|
||
do_b()
|
||
|
||
# Prints "Doing something"
|
||
```
|
||
|
||
Functions themselves no longer require formal definitions, either, they can be
|
||
created *[anonymously](http://en.wikipedia.org/wiki/Anonymous_function)*:
|
||
|
||
```python
|
||
my_send = lambda person, thing: send(person.upper(), thing, subject="Check this out!")
|
||
ucase_people = map(lambda name: name.upper(), ["Joe", "Mary", "Zach"])
|
||
```
|
||
|
||
|
||
### Abstracting behaviour
|
||
|
||
You'll find you can now start abstracting away common idioms. For
|
||
example, you probably very often find yourself looping over some list of items,
|
||
performing some set of actions on them, or passing them to some other function
|
||
or method:
|
||
|
||
```python
|
||
people = ["Joe", "Chris", "Matt", "Jennifer"]
|
||
for person in people:
|
||
u_person = person.upper()
|
||
send(person, super_fun_thing)
|
||
```
|
||
|
||
Instead of that, you could have a function that takes a list as one argument,
|
||
and a function to apply to each item in it as another:
|
||
|
||
```python
|
||
def dostuff(action, things):
|
||
result = []
|
||
for thing in things:
|
||
result.append(action(thing))
|
||
return result
|
||
|
||
dostuff(send, people)
|
||
```
|
||
|
||
The above example is actually just a simple definition of one of the most
|
||
common higher-order functions,
|
||
[map](http://docs.python.org/library/functions.html#map), which python already
|
||
provides for you. Another particularly useful higher-order function is
|
||
[filter](http://docs.python.org/library/functions.html#filter) which, given a
|
||
function that returns true of false if its criteria are met by the passed item,
|
||
will return the subset of the passed list that satisfy the filtering function:
|
||
|
||
```python
|
||
stuff = ["My notes.txt", "Matt's notes.txt", "My music.pls"]
|
||
my_stuff = filter(lambda s: s.startswith("My "), stuff)
|
||
|
||
# my_stuff = ["My notes.txt", "My music.pls"]
|
||
```
|
||
|
||
[List comprehensions](http://docs.python.org/tutorial/datastructures.html#list-comprehensions)
|
||
provide a cleaner, easier to read way to perform mapping and/or filtering on a
|
||
list:
|
||
|
||
```python
|
||
stuff = ["My notes.txt", "Matt's notes.txt", "My music.pls"]
|
||
|
||
my_stuff = [file for file in stuff if file.startswith("My ")]
|
||
# ["My notes.txt", "My music.pls"]
|
||
|
||
upper_stuff = [file.upper() for file in stuff]
|
||
# ["MY NOTES.TXT", "MATT'S NOTES.TXT", "MY MUSIC.PLS"]
|
||
|
||
music = [file.upper() for file in stuff if file.endswith(".pls")]
|
||
# ["MY MUSIC.PLS"]
|
||
```
|
||
|
||
|
||
### Tip of the iceberg
|
||
|
||
This is just a very small taste of functional programming concepts. Later, I'll
|
||
introduce a couple of functional languages, and explain what sets them apart
|
||
from object-oriented and imperative programming languages.
|
||
#+end_export
|
||
* DONE Erlang: The Movie :programming:erlang:
|
||
CLOSED: [2013-11-27 Wed]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: erlang-the-movie
|
||
:EXPORT_HUGO_SLUG: erlang-the-movie
|
||
:END:
|
||
|
||
#+begin_export markdown
|
||
Hopping through [Joe Armstrong's blog](http://joearms.github.io/), I happened across Erlang: The Movie.
|
||
More programming languages need videos like this.
|
||
|
||
<iframe width="420" height="315" src="//www.youtube.com/embed/xrIjfIjssLE" frameborder="0" allowfullscreen></iframe>
|
||
#+end_export
|
||
* DONE Getting Organized with Org Mode :emacs:org_mode:git:graphviz:
|
||
CLOSED: [2014-11-25 Tue]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: getting-organized-with-org-mode
|
||
:EXPORT_HUGO_SLUG: getting-organized-with-org-mode
|
||
:END:
|
||
|
||
#+begin_export markdown
|
||
<img src="/images/org-mode-unicorn-logo.png" alt="Org Mode logo" style="float: right" />
|
||
#+end_export
|
||
|
||
I've been using Emacs Org mode for nearly a year now. For a while I
|
||
mostly just used it to take and organize notes, but over time I've
|
||
discovered it's an incredibly useful tool for managing projects and
|
||
tasks, writing and publishing documents, keeping track of time and
|
||
todo lists, and maintaining a journal.
|
||
|
||
** Project Management
|
||
Most of what I've been using [[http://orgmode.org/][Org mode]] for has been breaking down large
|
||
projects at work into tasks and subtasks. It's really easy to enter
|
||
projects in as a hierarchy of tasks and task groupings. Using
|
||
[[http://orgmode.org/worg/org-tutorials/org-column-view-tutorial.html][Column View]], I was able to dive right into scoping them individually
|
||
and reporting total estimates for each major segment of work.
|
||
|
||
#+ATTR_HTML: :alt Example projects org file
|
||
[[file:static/images/emacs-projects.png]]
|
||
|
||
Because Org Mode makes building and modifying an outline structure
|
||
like this so quick and easy, I usually build and modify the project
|
||
org document while planning it out with my team. Once done, I then
|
||
manually load that information into our issue tracker and get
|
||
underway. Occasionally I'll also update tags and progress status in
|
||
the org document as well as the project progresses, so I can use the
|
||
same document to plan subsequent development iterations.
|
||
** Organizing Notes and Code Exercises
|
||
More recently, I've been looking into various ways to get more
|
||
things organized with Org mode. I've been stepping through
|
||
[[http://sarabander.github.io/sicp/][Structure and Interpretation of Computer Programs]] with some other
|
||
folks from work, and discovered that Org mode was an ideal fit for
|
||
keeping my notes and exercise work together. The latter is neatly
|
||
managed by [[http://orgmode.org/worg/org-contrib/babel/intro.html][Babel]], which let me embed and edit source examples and
|
||
my excercise solutions right in the org document itself, and even
|
||
export them to one or more scheme files to load into my
|
||
interpreter.
|
||
** Exporting and Publishing Documents
|
||
Publishing my notes with org is also a breeze. I've published
|
||
project plans and proposals to PDF to share with colleagues, and
|
||
exported my [[https://github.com/correl/sicp][SICP notes]] to html and [[http://sicp.phoenixinquis.net/][dropped them into a site]] built
|
||
with [[http://jekyllrb.com/][Jekyll]]. Embedding graphs and diagrams into exported documents
|
||
using [[http://www.graphviz.org/][Graphviz]], [[http://www.mcternan.me.uk/mscgen/][Mscgen]], and [[http://plantuml.sourceforge.net/][PlantUML]] has also really helped with
|
||
putting together some great project plans and documentation. A lot of
|
||
great examples using those tools (and more!) can be found [[http://home.fnal.gov/~neilsen/notebook/orgExamples/org-examples.html][here]].
|
||
** Emacs Configuration
|
||
While learning all the cool things I could do with Org mode and Babel,
|
||
it was only natural I'd end up using it to reorganize my [[https://github.com/correl/dotfiles/tree/master/.emacs.d][Emacs
|
||
configuration]]. Up until that point, I'd been managing my configuration
|
||
in a single init.el file, plus a directory full of mode or
|
||
purpose-specific elisp files that I'd loop through and load. Inspired
|
||
primarily by the blog post, [[http://zeekat.nl/articles/making-emacs-work-for-me.html]["Making Emacs Work For Me"]], and later by
|
||
others such as [[http://pages.sachachua.com/.emacs.d/Sacha.html][Sacha Chua's Emacs configuration]], I got all my configs
|
||
neatly organized into a single org file that gets loaded on
|
||
startup. I've found it makes it far easier to keep track of what I've
|
||
got configured, and gives me a reason to document and organize things
|
||
neatly now that it's living a double life as a [[https://github.com/correl/dotfiles/blob/master/.emacs.d/emacs.org][published document]] on
|
||
GitHub. I've still got a directory lying around with autoloaded
|
||
scripts, but now it's simply reserved for [[https://github.com/correl/dotfiles/blob/master/.emacs.d/emacs.org#auto-loading-elisp-files][tinkering and sensitive
|
||
configuration]].
|
||
** Tracking Habits
|
||
Another great feature of Org mode that I've been taking advantage
|
||
of a lot more lately is the [[http://orgmode.org/manual/Agenda-Views.html][Agenda]]. By defining some org files as
|
||
being agenda files, Org mode can examine these files for TODO
|
||
entries, scheduled tasks, deadlines and more to build out useful
|
||
agenda views to get a quick handle on what needs to be done and
|
||
when. While at first I started by simply syncing down my google
|
||
calendars as org-files (using [[http://orgmode.org/worg/code/awk/ical2org.awk][ical2org.awk]]), I've started
|
||
managing TODO lists in a dedicated org file. By adding tasks to
|
||
this file, scheduling them, and setting deadlines, I've been doing
|
||
a much better job of keeping track of things I need to get done
|
||
and (even more importantly) /when/ I need to get them done.
|
||
|
||
#+ATTR_HTML: :alt Agenda view snippet
|
||
[[file:static/images/emacs-org-agenda.png]]
|
||
|
||
This works not only for one-shot tasks, but also [[http://orgmode.org/manual/Tracking-your-habits.html][habits and other
|
||
repetitive tasks]]. It's possible to schedule a task that should be
|
||
done every day, every few days, or maybe every first sunday of a
|
||
month. For example, I've set up repeating tasks to write a blog
|
||
post at least once a month, practice guitar every two to three
|
||
days, and to do the dishes every one or two days. The agenda view
|
||
can even show a small, colorized graph next to each repeating task
|
||
that paints a picture of how well (or not!) I've been getting
|
||
those tasks done on time.
|
||
** Keeping a Journal and Tracking Work
|
||
The last thing I've been using (which I'm still getting a handle
|
||
on) is using [[http://orgmode.org/manual/Capture.html][Capture]] to take and store notes, keep a journal, and
|
||
even [[http://orgmode.org/manual/Clocking-work-time.html][track time on tasks at work]].
|
||
|
||
#+BEGIN_SRC emacs-lisp :exports code
|
||
(setq org-capture-templates
|
||
'(("j" "Journal Entry" plain
|
||
(file+datetree "~/org/journal.org")
|
||
"%U\n\n%?" :empty-lines-before 1)
|
||
("w" "Log Work Task" entry
|
||
(file+datetree "~/org/worklog.org")
|
||
"* TODO %^{Description} %^g\n%?\n\nAdded: %U"
|
||
:clock-in t
|
||
:clock-keep t)))
|
||
|
||
(global-set-key (kbd "C-c c") 'org-capture)
|
||
|
||
(setq org-clock-persist 'history)
|
||
(org-clock-persistence-insinuate)
|
||
#+END_SRC
|
||
|
||
For my journal, I've configured a capture template that I can use
|
||
to write down a new entry that will be stored with a time stamp
|
||
appended into its own org file, organized under headlines by year,
|
||
month and date.
|
||
|
||
For work tasks, I have another capture template configured that
|
||
will log and tag a task into another org file, also organized by
|
||
date, which will automatically start tracking time for that
|
||
task. Once done, I can simply clock out and check the time I've
|
||
spent, and can easily find it later to clock in again, add notes,
|
||
or update its status. This helps me keep track of what I've gotten
|
||
done during the day, keep notes on what I was doing at any point
|
||
in time, and get a better idea of how long it takes me to do
|
||
different types of tasks.
|
||
** Conclusion
|
||
There's a lot that can be done with Org mode, and I've only just
|
||
scratched the surface. The simple outline format provided by Org mode
|
||
lends itself to doing all sorts of things, be it organizing notes,
|
||
keeping a private or work journal, or writing a book or technical
|
||
document. I've even written this blog post in Org mode! There's tons
|
||
of functionality that can be built on top of it, yet the underlying
|
||
format itself remains simple and easy to work with. I've never been
|
||
great at keeping myself organized, but Org mode is such a delight to
|
||
use that I can't help trying anyway. If it can work for me, maybe it
|
||
can work for you, too!
|
||
|
||
There's tons of resources for finding new ways for using Org mode, and
|
||
I'm still discovering cool things I can track and integrate with it. I
|
||
definitely recommend reading through [[http://sachachua.com/blog/][Sacha Chua's Blog]], as well as
|
||
posts from [[http://newartisans.com/2007/08/using-org-mode-as-a-day-planner/][John Wiegley]]. I'm always looking for more stuff to try
|
||
out. Feel free to drop me a line if you find or are using something
|
||
you think is cool or useful!
|
||
|
||
* DONE Adventuring Through SICP :programming:lisp:
|
||
CLOSED: [2015-01-01 Thu]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: sicp
|
||
:EXPORT_HUGO_SLUG: sicp
|
||
:END:
|
||
|
||
Back in May, a coworker and I got the idea to start up a little
|
||
seminar after work every couple of weeks with the plan to set aside
|
||
some time to learn and discuss new ideas together, along with anyone
|
||
else who cared to join us.
|
||
|
||
** Learning Together
|
||
Over the past several months, we've read our way through the first
|
||
three chapters of the book, watched the [[http://ocw.mit.edu/courses/electrical-engineering-and-computer-science/6-001-structure-and-interpretation-of-computer-programs-spring-2005/video-lectures/][related video lectures]], and
|
||
did (most of) the exercises.
|
||
|
||
Aside from being a great excuse to unwind with friends after work
|
||
(which it is!), it's proved to be a great way to get through the
|
||
material. Doing a section of a chapter every couple of weeks is an
|
||
easy goal to meet, and meeting up to discuss it becomes something to
|
||
look forward to. We all get to enjoy a sense of accomplishment in
|
||
learning stuff that can be daunting or difficult to set aside time for
|
||
alone.
|
||
|
||
The best part, by far, is getting different perspectives on the
|
||
material. Most of my learning tends to be solitary, so it's refreshing
|
||
to do it with a group. By reviewing the different concepts together,
|
||
we're able to gain insights and clarity we'd never manage on our
|
||
own. Even the simplest topics can spur interesting conversations.
|
||
|
||
** SICP
|
||
Our first adventure together so far has been the venerable [[http://mitpress.mit.edu/sicp/][Structure
|
||
and Interpretation of Computer Programs]]. This book had been on my todo
|
||
list for a long time, but never quite bubbled to the top. I'm glad to
|
||
have the opportunity to go through it in this format, since there's
|
||
plenty of time to let really get into the excercises and let the
|
||
lessons sink in.
|
||
|
||
SICP was originally an introductory textbook for MIT computer
|
||
programming courses. What sets it apart from most, though, is that it
|
||
doesn't focus so much on learning a particular programming language
|
||
(while the book does use and cover MIT Scheme) as it does on
|
||
identifying and abstracting out patterns common to most programming
|
||
problems. Because of that, the book is every bit as useful and
|
||
illuminating as ever, especially now that functional paradigms are
|
||
re-entering the spotlight and means of abstracting and composing
|
||
systems are as important as ever.
|
||
|
||
** What's next?
|
||
We've still got plenty of SICP left to get through. We've only just
|
||
gotten through Chapter 4, section 1, which has us building a scheme
|
||
interpreter *in* scheme, so there's plenty of fun left to be had
|
||
there.
|
||
|
||
We're also staring to do some smaller, lunchtime review meetings
|
||
following the evening discussions to catch up the folks that can't
|
||
make it. I may also try sneaking in some smaller material, like
|
||
interesting blog posts, to keep things lively.
|
||
|
||
----------------------------------------------------------------------
|
||
|
||
If anyone's interested, I have the exercise work along with some notes
|
||
taken during the meetings [[http://sicp.phoenixinquis.net/][hosted online]]. I apologize for the lack of
|
||
notes early on, I've been trying to get better at capturing memorable
|
||
excerpts and conversation topics recently. I may have to put some more
|
||
posts together later on summarizing what we discussed for each
|
||
chapter; if and when I do, they'll be posted on the [[http://extreme-tech-seminar.github.io/][seminar website]].
|
||
|
||
* DONE Coders at Work :programming:books:
|
||
CLOSED: [2015-01-28 Wed]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: coders-at-work
|
||
:EXPORT_HUGO_SLUG: coders-at-work
|
||
:END:
|
||
# Gather highlights from the book and write a post summarizing my
|
||
# thoughts on it, and what I took away from it.
|
||
|
||
A few days before leaving work for a week and a half of flying and
|
||
cruising to escape frigid Pennsylvania, I came across a [[armstrong-oop][Joe Armstrong
|
||
quote]] during my regularly scheduled slacking off on twitter and Hacker
|
||
News. I'd come across a couple times before, only this time I noticed
|
||
it had a source link. This led me to discovering (and shortly
|
||
thereafter, buying) Peter Seibel's "[[http://www.codersatwork.com/][Coders at Work -- Reflections on
|
||
the Craft of Programming]]". I loaded it onto my nook, and off I went.
|
||
|
||
The book is essentially a collection of interviews with a series of
|
||
highly accomplished software developers. Each of them has their own
|
||
fascinating insights into the craft and its rich history.
|
||
|
||
While making my way through the book, I highlighted some excerpts
|
||
that, for one reason or another, resonated with me. I've organized and
|
||
elaborated on them below.
|
||
|
||
** DONE Incremental Changes
|
||
CLOSED: [2015-01-20 Tue 20:59]
|
||
<<fitzpatrick-increments>>
|
||
#+BEGIN_QUOTE
|
||
I've seen young programmers say, "Oh, shit, it doesn't work," and then
|
||
rewrite it all. Stop. Try to figure out what's going on. *Learn how to
|
||
write things incrementally so that at each stage you could verify it.*\\
|
||
-- Brad Fitzpatrick
|
||
#+END_QUOTE
|
||
|
||
I can remember doing this to myself when I was still relatively new to
|
||
coding (and even worse, before I discovered source control!). Some
|
||
subroutine or other would be misbehaving, and rather than picking it
|
||
apart and figuring out what it was I'd done wrong, I'd just blow it
|
||
away and attempt to write it fresh. While I /might/ be successful,
|
||
that likely depended on the issue being some sort of typo or missed
|
||
logic; if it was broken because I misunderstood something or had a bad
|
||
plan to begin with, rewriting it would only result in more broken
|
||
code, sometimes in more or different ways than before. I don't think
|
||
I've ever rewritten someone else's code without first at least getting
|
||
a firm understanding of it and what it was trying to accomplish, but
|
||
even then, breaking down changes piece by piece makes it all the
|
||
easier to maintain sanity.
|
||
|
||
I do still sometimes catch myself doing too much at once when building
|
||
a new feature or fixing a bug. I may have to fix a separate bug that's
|
||
in my way, or I may have to make several different changes in various
|
||
parts of the code. If I'm not careful, things can get out of hand
|
||
pretty quickly, and before I know it I have a blob of changes strewn
|
||
across the codebase in my working directory without a clear picture of
|
||
what's what. If something goes wrong, it can be pretty tough to sort
|
||
out which change broke things (or fixed them). Committing changes
|
||
often helps tremendously to avoid this sort of situation, and when I
|
||
catch myself going off the rails I try to find a stopping point and
|
||
split changes up into commits as soon as possible to regain
|
||
control. Related changes and fixes can always be squashed together
|
||
afterwards to keep things tidy.
|
||
|
||
** DONE Specifications & Documentation
|
||
CLOSED: [2015-01-20 Tue 20:59]
|
||
<<bloch-customers>>
|
||
#+BEGIN_QUOTE
|
||
*Many customers won't tell you a problem; they'll tell you a
|
||
solution.* A customer might say, for instance, "I need you to add
|
||
support for the following 17 attributes to this system. Then you have
|
||
to ask, 'Why? What are you going to do with the system? How do you
|
||
expect it to evolve?'" And so on. You go back and forth until you
|
||
figure out what all the customer really needs the software to
|
||
do. These are the use cases.\\
|
||
-- Joshua Bloch
|
||
#+END_QUOTE
|
||
|
||
Whether your customer is your customer, or your CEO, the point stands:
|
||
customers are /really bad/ at expressing what they want. It's hard to
|
||
blame them, though; analyzing what you really want and distilling it
|
||
into a clear specification is tough work. If your customer is your
|
||
boss, it can be intimidating to push back with questions like "Why?",
|
||
but if you can get those questions answered you'll end up with a
|
||
better product, a better /understanding/ of the product, and a happy
|
||
customer. The agile process of doing quick iterations to get tangible
|
||
results in front of them is a great way of getting the feedback and
|
||
answers you need.
|
||
|
||
<<armstrong-documentation>>
|
||
#+BEGIN_QUOTE
|
||
The code shows me what it /does/. It doesn't show me what it's
|
||
supposed to do. I think the code is the answer to a problem.
|
||
*If you don't have the spec or you don't have any documentation, you have to guess what the problem is from the answer. You might guess wrong.*\\
|
||
-- Joe Armstrong
|
||
#+END_QUOTE
|
||
|
||
Once you've got the definition of what you've got to build and how
|
||
it's got to work, it's extremely important that you get it
|
||
documented. Too often, I'm faced with code that's doing something in
|
||
some way that somebody, either a customer or a developer reading it,
|
||
takes issue with, and there's no documentation anywhere on why it's
|
||
doing what it's doing. What happens next is anybody's guess. Code
|
||
that's clear and conveys its intent is a good start towards avoiding
|
||
this sort of situation. Comments explaining intent help too, though
|
||
making sure they're kept up to date with the code can be
|
||
challenging. At the very least, I try to promote useful commit
|
||
messages explaining what the purpose of a change is, and reference a
|
||
ticket in our issue tracker which (hopefully) has a clear accounting
|
||
of the feature or bugfix that prompted it.
|
||
** DONE Pair Programming
|
||
CLOSED: [2015-01-20 Tue 21:03]
|
||
<<armstrong-pairing>>
|
||
#+BEGIN_QUOTE
|
||
... *if you don't know what you're doing then I think it can be very
|
||
helpful with someone who also doesn't know what they're doing.* If you
|
||
have one programmer who's better than the other one, then there's
|
||
probably benefit for the weaker programmer or the less-experienced
|
||
programmer to observe the other one. They're going to learn something
|
||
from that. But if the gap's too great then they won't learn, they'll
|
||
just sit there feeling stupid.\\
|
||
-- Joe Armstrong
|
||
#+END_QUOTE
|
||
|
||
Pairing isn't something I do much. At least, it's pretty rare that I
|
||
have someone sitting next to me as I code. I *do* involve peers while
|
||
I'm figuring out what I want to build as often as I can. The tougher
|
||
the problem, the more important it is, I think, to get as much
|
||
feedback and brainstorming in as possible. This way, everybody gets to
|
||
tackle the problem and learn together, and anyone's input, however
|
||
small it might seem, can be the key to the "a-ha" moment to figuring
|
||
out a solution.
|
||
|
||
** DONE Peer Review
|
||
CLOSED: [2015-01-25 Sun 22:44]
|
||
<<crockford-reading>>
|
||
#+BEGIN_QUOTE
|
||
*I think an hour of code reading is worth two weeks of QA.* It's just
|
||
a really effective way of removing errors. If you have someone who is
|
||
strong reading, then the novices around them are going to learn a lot
|
||
that they wouldn't be learning otherwise, and if you have a novice
|
||
reading, he's going to get a lot of really good advice.\\
|
||
-- Douglas Crockford
|
||
#+END_QUOTE
|
||
|
||
Just as important as designing the software as a team, I think, is
|
||
reviewing it as a team. In doing so, each member of the team has an
|
||
opportunity to understand /how/ the system has been implemented, and
|
||
to offer their suggestions and constructive criticisms. This helps the
|
||
team grow together, and results in a higher quality of code overall.
|
||
This benefits QA as well as the developers themselves for the next
|
||
time they find themselves in that particular bit of the system.
|
||
** DONE Object-Oriented Programming
|
||
CLOSED: [2015-01-20 Tue 20:59]
|
||
<<armstrong-oop>>
|
||
#+BEGIN_QUOTE
|
||
I think the lack of reusability comes in object-oriented languages,
|
||
not in functional languages.
|
||
*Because the problem with object-oriented languages is they've got all this implicit environment that they carry around with them. You wanted a banana but what you got was a gorilla holding the banana and the entire jungle.*\\
|
||
-- Joe Armstrong
|
||
#+END_QUOTE
|
||
|
||
A lot has been written on why OOP isn't the great thing it claims to
|
||
be, or was ever intended to be. Having grappled with it myself for
|
||
years, attempting to find ways to keep my code clean, concise and
|
||
extensible, I've more or less come to the same conclusion as Armstrong
|
||
in that coupling data structures with behaviour makes for a terrible
|
||
mess. Dividing the two led to a sort of moment of clarity; there was
|
||
no more confusion about what methods belong on what object. There was
|
||
simply the data, and the methods that act on it. I am still struggling
|
||
a bit, though, on how to bring this mindset to the PHP I maintain at
|
||
work. The language seems particularly ill-suited to managing complex
|
||
data structures (or even simple ones -- vectors and hashes are
|
||
bizarrely intertwined).
|
||
** DONE Writing
|
||
CLOSED: [2015-01-28 Wed 22:42]
|
||
<<bloch-writing>>
|
||
#+BEGIN_QUOTE
|
||
You should read /[Elements of Style]/ for two reasons: The first is
|
||
that a large part of every software engineer's job is writing
|
||
prose. *If you can't write precise, coherent, readable specs, nobody
|
||
is going to be able to use your stuff.* So anything that improves your
|
||
prose style is good. The second reason is that most of the ideas in
|
||
that book are also applicable to programs.\\
|
||
-- Joshua Bloch
|
||
#+END_QUOTE
|
||
|
||
<<crockford-writing>>
|
||
#+BEGIN_QUOTE
|
||
*My advice to everybody is pretty much the same, to read and write.*\\
|
||
...\\
|
||
Are you a good Java programmer, a good C programmer, or whatever? I
|
||
don't care. I just want to know that you know how to put an algorithm
|
||
together, you understand data structures, and you know how to document
|
||
it.\\
|
||
-- Douglas Crockford
|
||
#+END_QUOTE
|
||
|
||
<<knuth-writing>>
|
||
#+BEGIN_QUOTE
|
||
This is what literate programming is so great for --\\
|
||
*I can talk to myself. I can read my program a year later and know
|
||
exactly what I was thinking.*\\
|
||
-- Donald Knuth
|
||
#+END_QUOTE
|
||
|
||
The more I've program professionally, the clearer it is that writing
|
||
(and communication in general) is a very important skill to
|
||
develop. Whether it be writing documentation, putting together a
|
||
project plan, or whiteboarding and discussing something, clear and
|
||
concise communication skills are a must. Clarity in writing translates
|
||
into clarity in coding as well, in my opinion. Code that is short, to
|
||
the point, clear in its intention, making good use of structure and
|
||
wording (in the form of function and variable names) is far easier to
|
||
read and reason about than code that is disorganized and obtuse.
|
||
** DONE Knuth
|
||
CLOSED: [2015-01-28 Wed 22:42]
|
||
<<crockford-knuth>>
|
||
#+BEGIN_QUOTE
|
||
I tried to make familiarity with Knuth a hiring criteria, and I was
|
||
disappointed that I couldn't find enough people that had read him. In
|
||
my view,
|
||
*anybody who calls himself a professional programmer should have read
|
||
Knuth's books or at least should have copies of his books.*\\
|
||
-- Douglas Crockford
|
||
#+END_QUOTE
|
||
|
||
<<steele-knuth>>
|
||
#+BEGIN_QUOTE
|
||
... Knuth is really good at telling a story about code. When you read
|
||
your way through /The Art of Computer Programming/ and you read your
|
||
way through an algorithm, he's explained it to you and showed you some
|
||
applications and given you some exercises to work, and *you feel like
|
||
you've been led on a worthwhile journey.*\\
|
||
-- Guy Steele
|
||
#+END_QUOTE
|
||
|
||
<<norvig-knuth>>
|
||
#+BEGIN_QUOTE
|
||
At one point I had /[The Art of Computer Programming]/ as my monitor
|
||
stand because it was one of the biggest set of books I had, and it was
|
||
just the right height. That was nice because it was always there, and
|
||
I guess then I was more prone to use it as a reference because it was
|
||
right in front of me.\\
|
||
-- Peter Norvig
|
||
#+END_QUOTE
|
||
|
||
I haven't read any of Knuth's books yet, which is something I'll have
|
||
to rectify soon. I don't think I have the mathematical background
|
||
necessary to get through some of his stuff, but I expect it will be
|
||
rewarding nonetheless. I'm also intrigued by his concept of literate
|
||
programming, and I'm curious to learn more about TeX. I imagine I'll
|
||
be skimming through [[http://brokestream.com/tex-web.html][TeX: The Program]] pretty soon now that I've
|
||
finished Coders at Work :)
|
||
|
||
* DONE Birthday Puzzle :programming:prolog:
|
||
CLOSED: [2015-04-18 Sat]
|
||
:PROPERTIES:
|
||
:header-args:prolog: :system swipl :session *birthday* :goal true :exports both
|
||
:EXPORT_FILE_NAME: birthday-puzzle
|
||
:EXPORT_HUGO_SLUG: birthday-puzzle
|
||
:END:
|
||
|
||
This logic puzzle has been floating around the internet lately. When I
|
||
caught wind of it, I thought it would be a great exercise to tackle
|
||
using Prolog. I'm not especially good with the language yet, so it
|
||
added to the challenge a bit, but it was a pretty worthwhile
|
||
undertaking. When I got stumped, I discovered that mapping out the
|
||
birthdays into a grid helped me visualize the problem and ultimately
|
||
solve it, so I've included that with my prolog code so you can see how
|
||
I arrived at the answer.
|
||
|
||
** The Puzzle
|
||
Albert and Bernard have just met Cheryl. “When is your birthday?”
|
||
Albert asked Cheryl. Cheryl thought for a moment and said, “I won’t
|
||
tell you, but I’ll give you some clues”. She wrote down a list of
|
||
ten dates:
|
||
|
||
|
||
- May 15, May 16, May 19
|
||
- June 17, June 18
|
||
- July 14, July 16
|
||
- August 14, August 15, August 17
|
||
|
||
“One of these is my birthday,” she said.
|
||
|
||
|
||
Cheryl whispered in Albert’s ear the month, and only the month, of
|
||
her birthday. To Bernard, she whispered the day, and only the
|
||
day. “Can you figure it out now?” she asked Albert.
|
||
|
||
|
||
Albert: “I don’t know when your birthday is, but I know Bernard
|
||
doesn’t know, either.”
|
||
|
||
Bernard: “I didn’t know originally, but now I do.”
|
||
|
||
Albert: “Well, now I know, too!”
|
||
|
||
/When is Cheryl’s birthday?/
|
||
|
||
** The Solution
|
||
*** The Dates
|
||
|
||
To start off, i entered each of the possible birthdays as facts:
|
||
|
||
#+BEGIN_SRC prolog :results silent
|
||
possible_birthday(may, 15).
|
||
possible_birthday(may, 16).
|
||
possible_birthday(may, 19).
|
||
possible_birthday(june, 17).
|
||
possible_birthday(june, 18).
|
||
possible_birthday(july, 14).
|
||
possible_birthday(july, 16).
|
||
possible_birthday(august, 14).
|
||
possible_birthday(august, 15).
|
||
possible_birthday(august, 17).
|
||
#+END_SRC
|
||
|
||
And here they are, mapped out in a grid:
|
||
|
||
| | <c> | <c> | <c> | <c> |
|
||
| | May | June | July | August |
|
||
|----+-----+------+------+--------|
|
||
| 14 | | | X | X |
|
||
| 15 | X | | | X |
|
||
| 16 | X | | X | |
|
||
| 17 | | X | | X |
|
||
| 18 | | X | | |
|
||
| 19 | X | | | |
|
||
|
||
*** Albert's Statement
|
||
#+BEGIN_QUOTE
|
||
I don’t know when your birthday is,...
|
||
#+END_QUOTE
|
||
|
||
Albert only knows the month, and the month isn't enough to uniquely
|
||
identify Cheryl's birthday.
|
||
|
||
#+BEGIN_SRC prolog :results silent
|
||
month_is_not_unique(M) :-
|
||
bagof(D, possible_birthday(M, D), Days),
|
||
length(Days, Len),
|
||
Len > 1.
|
||
#+END_SRC
|
||
|
||
#+BEGIN_QUOTE
|
||
... but I know Bernard doesn’t know, either.
|
||
#+END_QUOTE
|
||
|
||
Albert knows that Bernard doesn't know Cheryl's
|
||
birthday. Therefore, the day alone isn't enough to know Cheryl's
|
||
birthday, and we can infer that the month of Cheryl's birthday does
|
||
not include any of the unique dates.
|
||
|
||
#+BEGIN_SRC prolog :results silent
|
||
day_is_not_unique(D) :-
|
||
bagof(M, possible_birthday(M, D), Months),
|
||
length(Months, Len),
|
||
Len > 1.
|
||
|
||
month_has_no_unique_days(M) :-
|
||
forall(possible_birthday(M,D),
|
||
day_is_not_unique(D)).
|
||
#+END_SRC
|
||
|
||
Based on what Albert knows at this point, let's see how we've
|
||
reduced the possible dates:
|
||
|
||
#+HEADER: :goal findall((M,D), part_one(M,D), Results)
|
||
#+BEGIN_SRC prolog
|
||
part_one(M,D) :-
|
||
possible_birthday(M,D),
|
||
month_is_not_unique(M),
|
||
month_has_no_unique_days(M),
|
||
day_is_not_unique(D).
|
||
#+END_SRC
|
||
|
||
#+RESULTS:
|
||
: Results = [ (july, 14), (july, 16), (august, 14), (august, 15), (august, 17)].
|
||
|
||
So the unique days (the 18th and 19th) are out, as are the months
|
||
that contained them (May and June).
|
||
|
||
| | <c> | <c> |
|
||
| | July | August |
|
||
|----+------+--------|
|
||
| 14 | X | X |
|
||
| 15 | | X |
|
||
| 16 | X | |
|
||
| 17 | | X |
|
||
|
||
*** Bernard's Statement
|
||
#+BEGIN_QUOTE
|
||
I didn’t know originally, but now I do.
|
||
#+END_QUOTE
|
||
|
||
For Bernard to know Cheryl's birthday, the day he knows must be
|
||
unique within the constraints we have so far.
|
||
|
||
#+BEGIN_SRC prolog :goal findall((M,D), part_two(M,D), Results)
|
||
day_is_unique(Month, Day) :-
|
||
findall(M, part_one(M, Day), [Month]).
|
||
part_two(Month, Day) :-
|
||
possible_birthday(Month, Day),
|
||
day_is_unique(Month, Day).
|
||
#+END_SRC
|
||
|
||
#+RESULTS:
|
||
: Results = [ (july, 16), (august, 15), (august, 17)].
|
||
|
||
Both July and August contain the 14th, so that row is out.
|
||
|
||
| | July | August |
|
||
|----+------+--------|
|
||
| 15 | | X |
|
||
| 16 | X | |
|
||
| 17 | | X |
|
||
|
||
*** Albert's Second Statement
|
||
#+BEGIN_QUOTE
|
||
Well, now I know, too!
|
||
#+END_QUOTE
|
||
|
||
Albert's month must be the remaining unique month:
|
||
|
||
#+BEGIN_SRC prolog :goal findall((M,D), part_three(M,D), Results)
|
||
month_is_not_unique(Month, Day) :-
|
||
findall(D, part_two(Month, D), [Day]).
|
||
part_three(Month, Day) :-
|
||
possible_birthday(Month, Day),
|
||
month_is_not_unique(Month, Day).
|
||
#+END_SRC
|
||
|
||
#+RESULTS:
|
||
: Results = [ (july, 16)].
|
||
|
||
August had two possible days, so it's now clear that the only
|
||
possible unique answer is July 16th.
|
||
|
||
| | <c> |
|
||
| | July |
|
||
|----+------|
|
||
| 15 | |
|
||
| 16 | X |
|
||
| 17 | |
|
||
|
||
*** Cheryl's Birthday
|
||
#+BEGIN_SRC prolog :goal cheryls_birthday(Month, Day)
|
||
cheryls_birthday(Month, Day) :-
|
||
part_three(Month, Day).
|
||
#+END_SRC
|
||
|
||
#+RESULTS:
|
||
: Month = july,
|
||
: Day = 16.
|
||
|
||
So, there we have it. Cheryl's birthday is July 16th!
|
||
|
||
| | <c> |
|
||
| | July |
|
||
|----+------|
|
||
| 16 | X |
|
||
* DONE Keeping Files And Configuration In Sync :git:
|
||
CLOSED: [2015-04-20 Mon]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: syncing
|
||
:EXPORT_HUGO_SLUG: syncing
|
||
:END:
|
||
|
||
I have a few computers I use on a daily basis, and I like to keep the
|
||
same emacs and shell configuration on all of them, along with my org
|
||
files and a handful of scripts. Since I'm sure other people have this
|
||
problem as well, I'll share what I'm doing so anyone can learn from
|
||
(or criticise) my solutions.
|
||
|
||
** Git for configuration and projects
|
||
|
||
I'm a software developer, so keeping things in git just makes sense
|
||
to me. I keep my org files in a privately hosted git repository, and
|
||
[[https://www.gnu.org/software/emacs/][Emacs]] and [[http://www.zsh.org/][Zsh]] configurations in a [[https://github.com/correl/dotfiles][public repo on github]]. My blog is
|
||
also hosted and published on github as well; I like having it cloned
|
||
to all my machines so I can work on drafts wherever I may be.
|
||
|
||
My [[https://github.com/correl/dotfiles/blob/master/.zshrc][.zshrc]] installs [[https://github.com/robbyrussell/oh-my-zsh][oh-my-zsh]] if it isn't installed already, and sets
|
||
up my shell theme, path, and some other environmental things.
|
||
|
||
My [[https://github.com/correl/dotfiles/blob/master/.emacs.d/emacs.org][Emacs configuration]] behaves similarly, making use of John
|
||
Wiegley's excellent [[https://github.com/jwiegley/use-package][use-package]] tool to ensure all my packages are
|
||
installed if they're not already there and configured the way I like
|
||
them.
|
||
|
||
All I have to do to get running on a new system is to install git,
|
||
emacs and zsh, clone my repo, symlink the files, and grab a cup of
|
||
tea while everything installs.
|
||
|
||
** Bittorrent sync for personal settings & books
|
||
|
||
For personal configuration that doesn't belong in and/or is too
|
||
sensitive to be in a public repo, I have a folder of dotfiles and
|
||
things that I sync between my machines using [[https://www.getsync.com/][Bittorrent Sync]]. The
|
||
dotfiles are arranged into directories by their purpose:
|
||
|
||
#+BEGIN_EXAMPLE
|
||
[correlr@reason:~/dotenv]
|
||
% tree -a -L 2
|
||
.
|
||
├── authinfo
|
||
│ └── .authinfo.gpg
|
||
├── bin
|
||
│ └── .bin
|
||
├── emacs
|
||
│ ├── .bbdb
|
||
│ └── .emacs.local.d
|
||
├── mail
|
||
│ ├── .gnus.el
|
||
│ ├── .signature
|
||
├── README.org
|
||
├── .sync
|
||
│ ├── Archive
|
||
│ ├── ID
|
||
│ ├── IgnoreList
|
||
│ └── StreamsList
|
||
├── tex
|
||
│ └── texmf
|
||
├── xmonad
|
||
│ └── .xmonad
|
||
└── zsh
|
||
└── .zshenv
|
||
#+END_EXAMPLE
|
||
|
||
This folder structure allows my configs to be easily installed using
|
||
[[https://www.gnu.org/software/stow/][GNU Stow]] from my =dotenv= folder:
|
||
|
||
: stow -vvS *
|
||
|
||
Running that command will, for each file in each of the directories,
|
||
create a symlink to it in my home folder if there isn't a file or
|
||
directory with that name there already.
|
||
|
||
Bittorrent sync also comes in handy for syncing my growing [[http://calibre-ebook.com/][Calibre]] ebook
|
||
collection, which outgrew my [[https://www.dropbox.com/][Dropbox]] account a while back.
|
||
* DONE Drawing Git Graphs with Graphviz and Org-Mode :emacs:org_mode:git:graphviz:
|
||
CLOSED: [2015-07-12 Sun]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: git-graphs
|
||
:EXPORT_HUGO_SLUG: git-graphs
|
||
:header-args:emacs-lisp: :results silent
|
||
:header-args:dot: :exports both
|
||
:END:
|
||
|
||
#+begin_export html
|
||
<style type="text/css">
|
||
svg text {
|
||
fill: white;
|
||
}
|
||
svg path,
|
||
svg polygon,
|
||
svg ellipse {
|
||
stroke: white;
|
||
}
|
||
</style>
|
||
#+end_export
|
||
|
||
Digging through Derek Feichtinger's [[https://github.com/dfeich/org-babel-examples][org-babel examples]] (which I came
|
||
across via [[http://irreal.org/blog/?p%3D4162][irreal.org]]), I found he had some great examples of
|
||
displaying git-style graphs using graphviz. I thought it'd be a fun
|
||
exercise to generate my own graphs based on his graphviz source using
|
||
elisp, and point it at actual git repos.
|
||
|
||
** Getting Started
|
||
|
||
I started out with the goal of building a simple graph showing a
|
||
mainline branch and a topic branch forked from it and eventually
|
||
merged back in.
|
||
|
||
Using Derek's example as a template, I described 5 commits on a master
|
||
branch, plus two on a topic branch.
|
||
|
||
#+NAME: git-graphs-example
|
||
#+begin_src dot :file git-graphs-example.svg
|
||
digraph G {
|
||
rankdir="LR";
|
||
bgcolor="transparent";
|
||
node[width=0.15, height=0.15, shape=point, color=white];
|
||
edge[weight=2, arrowhead=none, color=white];
|
||
node[group=master];
|
||
1 -> 2 -> 3 -> 4 -> 5;
|
||
node[group=branch];
|
||
2 -> 6 -> 7 -> 4;
|
||
}
|
||
#+end_src
|
||
|
||
The resulting image looks like this:
|
||
#+RESULTS[a7cf21cb99be72abc22593af68f374b04297803c]: git-graphs-example
|
||
[[file:static/ox-hugo/git-graphs-example.svg]]
|
||
|
||
*** Designing the Data Structure
|
||
|
||
The first thing I needed to do was describe my data structure. Leaning
|
||
on my experiences reading and working through [[https://www.google.com/url?sa%3Dt&rct%3Dj&q%3D&esrc%3Ds&source%3Dweb&cd%3D1&cad%3Drja&uact%3D8&ved%3D0CB8QFjAA&url%3Dhttps%253A%252F%252Fmitpress.mit.edu%252Fsicp%252F&ei%3DlH6gVau5OIGR-AG8j7yACQ&usg%3DAFQjCNHTCXQK7qN-kYibdy_MqRBWxlr8og&sig2%3DLu9WIhyuTJS92e8hxne0Aw&bvm%3Dbv.97653015,d.cWw][SICP]], I got to work
|
||
building a constructor function, and several accessors.
|
||
|
||
I decided to represent each node on a graph with an id, a list of
|
||
parent ids, and a group which will correspond to the branch on the
|
||
graph the commit belongs to.
|
||
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/make-node (id &optional parents group)
|
||
(list id parents group))
|
||
|
||
(defun git-graph/node-id (node)
|
||
(nth 0 node))
|
||
|
||
(defun git-graph/node-parents (node)
|
||
(nth 1 node))
|
||
|
||
(defun git-graph/node-group (node)
|
||
(nth 2 node))
|
||
#+end_src
|
||
|
||
*** Converting the structure to Graphviz
|
||
|
||
Now that I had my data structures sorted out, it was time to step
|
||
through them and generate the graphviz source that'd give me the
|
||
nice-looking graphs I was after.
|
||
|
||
The graph is constructed using the example above as a template. The
|
||
nodes are defined first, followed by the edges between them.
|
||
|
||
#+name: git-graph/to-graphviz
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/to-graphviz (id nodes)
|
||
(string-join
|
||
(list
|
||
(concat "digraph " id " {")
|
||
"bgcolor=\"transparent\";"
|
||
"rankdir=\"LR\";"
|
||
"node[width=0.15,height=0.15,shape=point,fontsize=8.0,color=white,fontcolor=white];"
|
||
"edge[weight=2,arrowhead=none,color=white];"
|
||
(string-join
|
||
(-map #'git-graph/to-graphviz-node nodes)
|
||
"\n")
|
||
(string-join
|
||
(-uniq (-flatten (-map
|
||
(lambda (node) (git-graph/to-graphviz-edges node nodes))
|
||
nodes)))
|
||
"\n")
|
||
"}")
|
||
"\n"))
|
||
#+end_src
|
||
|
||
For the sake of readability, I'll format the output:
|
||
|
||
#+name: git-graph/to-graphviz
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/to-graphviz-pretty (id nodes)
|
||
(with-temp-buffer
|
||
(graphviz-dot-mode)
|
||
(insert (git-graph/to-graphviz id nodes))
|
||
(indent-region (point-min) (point-max))
|
||
(buffer-string)))
|
||
#+end_src
|
||
|
||
|
||
Each node is built, setting its group attribute when applicable.
|
||
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/to-graphviz-node (node)
|
||
(let ((node-id (git-graph/to-graphviz-node-id
|
||
(git-graph/node-id node))))
|
||
(concat node-id
|
||
(--if-let (git-graph/node-group node)
|
||
(concat "[group=\"" it "\"]"))
|
||
";")))
|
||
#+end_src
|
||
|
||
Graphviz node identifiers are quoted to avoid running into issues with
|
||
spaces or other special characters.
|
||
|
||
#+name: git-graph/to-graphviz-nodes
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/to-graphviz-node-id (id)
|
||
(format "\"%s\"" id))
|
||
#+end_src
|
||
|
||
For each node, an edge is built connecting the node to each of its
|
||
parents.
|
||
|
||
#+name: git-graph/to-graphviz-edges
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/to-graphviz-edges (node &optional nodelist)
|
||
(let ((node-id (git-graph/node-id node))
|
||
(parents (git-graph/node-parents node))
|
||
(node-ids (-map #'git-graph/node-id nodelist)))
|
||
(-map (lambda (parent)
|
||
(unless (and nodelist (not (member parent node-ids)))
|
||
(git-graph/to-graphviz-edge node-id parent)))
|
||
parents)))
|
||
|
||
(defun git-graph/to-graphviz-edge (from to)
|
||
(concat
|
||
(git-graph/to-graphviz-node-id to)
|
||
" -> "
|
||
(git-graph/to-graphviz-node-id from)
|
||
";"))
|
||
#+end_src
|
||
|
||
With that done, the simple graph above could be generated with the
|
||
following code:
|
||
|
||
#+name: git-example
|
||
#+begin_src emacs-lisp :results silent
|
||
(git-graph/to-graphviz-pretty
|
||
"example"
|
||
(list (git-graph/make-node 1 nil "master")
|
||
(git-graph/make-node 2 '(1) "master")
|
||
(git-graph/make-node 3 '(2) "master")
|
||
(git-graph/make-node 4 '(3 7) "master")
|
||
(git-graph/make-node 5 '(4) "master")
|
||
(git-graph/make-node 6 '(2) "branch")
|
||
(git-graph/make-node 7 '(6) "branch")))
|
||
#+end_src
|
||
|
||
Which generates the following graphviz source:
|
||
|
||
#+NAME: git-graphs-generated-example
|
||
#+begin_src dot :noweb yes :file "git-graphs-generated-example.svg"
|
||
<<git-example()>>
|
||
#+end_src
|
||
|
||
The generated image matches the example exactly:
|
||
|
||
#+RESULTS[124faae6db8992b9cf42cabab4d1493f973aa6c5]: git-graphs-generated-example
|
||
[[file:static/ox-hugo/git-graphs-generated-example.svg]]
|
||
|
||
** Adding Labels
|
||
|
||
The next thing my graph needed was a way of labeling nodes. Rather
|
||
than trying to figure out some way of attaching a separate label to a
|
||
node, I decided to simply draw a labeled node as a box with text.
|
||
|
||
#+begin_src dot :file "git-graphs-labels.svg"
|
||
digraph G {
|
||
rankdir="LR";
|
||
bgcolor="transparent";
|
||
node[width=0.15, height=0.15, shape=point,fontsize=8.0,color=white,fontcolor=white];
|
||
edge[weight=2, arrowhead=none,color=white];
|
||
node[group=main];
|
||
1 -> 2 -> 3 -> 4 -> 5;
|
||
5[shape=box,label=master];
|
||
node[group=branch1];
|
||
2 -> 6 -> 7 -> 4;
|
||
7[shape=box,label=branch];
|
||
}
|
||
#+end_src
|
||
|
||
#+RESULTS[2d1e27579abf3bcd67093d101de7b9f6ec61eb52]:
|
||
[[file:static/ox-hugo/git-graphs-labels.svg]]
|
||
|
||
*** Updating the Data Structure
|
||
|
||
I updated my data structure to support an optional label applied to a
|
||
node. I opted to store it in an associative list alongside the group.
|
||
|
||
#+name: git-graph/structure
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/make-node (id &optional parents options)
|
||
(list id parents options))
|
||
|
||
(defun git-graph/node-id (node)
|
||
(nth 0 node))
|
||
|
||
(defun git-graph/node-parents (node)
|
||
(nth 1 node))
|
||
|
||
(defun git-graph/node-group (node)
|
||
(cdr (assoc 'group (nth 2 node))))
|
||
|
||
(defun git-graph/node-label (node)
|
||
(cdr (assoc 'label (nth 2 node))))
|
||
#+end_src
|
||
|
||
*** Updating the Graphviz node generation
|
||
|
||
The next step was updating the Graphviz generation functions to handle
|
||
the new data structure, and set the shape and label attributes of
|
||
labeled nodes.
|
||
|
||
#+name: git-graph/to-graphviz-nodes
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/to-graphviz-node (node)
|
||
(let ((node-id (git-graph/to-graphviz-node-id (git-graph/node-id node))))
|
||
(concat node-id
|
||
(git-graph/to-graphviz-node--attributes node)
|
||
";")))
|
||
|
||
(defun git-graph/to-graphviz-node--attributes (node)
|
||
(let ((attributes (git-graph/to-graphviz-node--compute-attributes node)))
|
||
(and attributes
|
||
(concat "["
|
||
(mapconcat (lambda (pair)
|
||
(format "%s=\"%s\""
|
||
(car pair) (cdr pair)))
|
||
attributes
|
||
", ")
|
||
"]"))))
|
||
|
||
(defun git-graph/to-graphviz-node--compute-attributes (node)
|
||
(-filter #'identity
|
||
(append (and (git-graph/node-group node)
|
||
(list (cons 'group (git-graph/node-group node))))
|
||
(and (git-graph/node-label node)
|
||
(list (cons 'shape 'box)
|
||
(cons 'label (git-graph/node-label node)))))))
|
||
#+end_src
|
||
|
||
I could then label the tips of each branch:
|
||
|
||
#+name: graph-example-labels
|
||
#+begin_src emacs-lisp :exports code :results silent
|
||
(git-graph/to-graphviz-pretty
|
||
"labeled"
|
||
(list (git-graph/make-node 1 nil '((group . "master")))
|
||
(git-graph/make-node 2 '(1) '((group . "master")))
|
||
(git-graph/make-node 3 '(2) '((group . "master")))
|
||
(git-graph/make-node 4 '(3 7) '((group . "master")))
|
||
(git-graph/make-node 5 '(4) '((group . "master")
|
||
(label . "master")))
|
||
(git-graph/make-node 6 '(2) '((group . "branch")))
|
||
(git-graph/make-node 7 '(6) '((group . "branch")
|
||
(label . "branch")))))
|
||
#+end_src
|
||
|
||
#+begin_src dot :file "git-graphs-labels-generated.svg" :noweb yes :exports results
|
||
<<graph-example-labels()>>
|
||
#+end_src
|
||
|
||
#+RESULTS[e5a194d1f4c737ff465c20d6b063ab58f9530a72]:
|
||
[[file:static/ox-hugo/git-graphs-labels-generated.svg]]
|
||
|
||
** Automatic Grouping Using Leaf Nodes
|
||
|
||
Manually assigning groups to each node is tedious, and easy to
|
||
accidentally get wrong. Also, with the goal to graph git repositories,
|
||
I was going to have to figure out groupings automatically anyway.
|
||
|
||
To do this, it made sense to traverse the nodes in [[https://en.wikipedia.org/wiki/Topological_sorting][topological order]].
|
||
|
||
Repeating the example above,
|
||
#+begin_src dot :file git-graphs-topo.svg
|
||
digraph G {
|
||
rankdir="LR";
|
||
bgcolor="transparent";
|
||
node[width=0.15, height=0.15, shape=circle, color=white, fontcolor=white];
|
||
edge[weight=2, arrowhead=none, color=white];
|
||
node[group=main];
|
||
1 -> 2 -> 3 -> 4 -> 5;
|
||
node[group=branch1];
|
||
2 -> 6 -> 7 -> 4;
|
||
}
|
||
#+end_src
|
||
|
||
#+RESULTS[277f98904b151a521fcdb45b5a77568f481639c1]:
|
||
[[file:static/ox-hugo/git-graphs-topo.svg]]
|
||
|
||
These nodes can be represented (right to left) in topological order as
|
||
either ~5, 4, 3, 7, 6, 2, 1~ or ~5, 4, 7, 6, 3, 2, 1~.
|
||
|
||
Having no further children, ~5~ is a leaf node, and can be used as a
|
||
group. All first parents of ~5~ can therefore be considered to be in
|
||
group ~5~.
|
||
|
||
~7~ is a second parent to ~4~, and so should be used as the group for
|
||
all of its parents not present in group ~5~.
|
||
|
||
#+name: git-graph/group-topo
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/group-topo (nodelist)
|
||
(reverse
|
||
(car
|
||
(-reduce-from
|
||
(lambda (acc node)
|
||
(let* ((grouped-nodes (car acc))
|
||
(group-stack (cdr acc))
|
||
(node-id (git-graph/node-id node))
|
||
(group-from-stack (--if-let (assoc node-id group-stack)
|
||
(cdr it)))
|
||
(group (or group-from-stack node-id))
|
||
(parents (git-graph/node-parents node))
|
||
(first-parent (first parents)))
|
||
(if group-from-stack
|
||
(pop group-stack))
|
||
(if (and first-parent (not (assoc first-parent group-stack)))
|
||
(push (cons first-parent group) group-stack))
|
||
(cons (cons (git-graph/make-node node-id
|
||
parents
|
||
`((group . ,group)
|
||
(label . ,(git-graph/node-label node))))
|
||
grouped-nodes)
|
||
group-stack)))
|
||
nil
|
||
nodelist))))
|
||
#+end_src
|
||
|
||
While iterating through the node list, I maintained a stack of pairs
|
||
built from the first parent of the current node, and the current
|
||
group. To determine the group, the head of the stack is checked to see
|
||
if it contains a group for the current node id. If it does, that group
|
||
is used and it is popped off the stack, otherwise the current node id
|
||
is used.
|
||
|
||
The following table illustrates how the stack is used to store and
|
||
assign group relationships as the process iterates through the node
|
||
list:
|
||
|
||
#+caption: Progressing through the nodes
|
||
| Node | Parents | Group Stack | Group |
|
||
|------+---------+-----------------+-------|
|
||
| 5 | (4) | (4 . 5) | 5 |
|
||
| 4 | (3 7) | (3 . 5) | 5 |
|
||
| 3 | (2) | (2 . 5) | 5 |
|
||
| 7 | (6) | (6 . 7) (2 . 5) | 7 |
|
||
| 6 | (2) | (2 . 5) | 7 |
|
||
| 2 | (1) | (1 . 5) | 5 |
|
||
| 1 | | | 5 |
|
||
|
||
|
||
*** Graph without automatic grouping
|
||
|
||
#+name: graph-no-auto-grouping
|
||
#+begin_src emacs-lisp :exports code :results silent
|
||
(git-graph/to-graphviz-pretty
|
||
"nogroups"
|
||
(list (git-graph/make-node 5 '(4) '((label . master)))
|
||
(git-graph/make-node 4 '(3 7))
|
||
(git-graph/make-node 3 '(2))
|
||
(git-graph/make-node 7 '(6) '((label . develop)))
|
||
(git-graph/make-node 6 '(2))
|
||
(git-graph/make-node 2 '(1))
|
||
(git-graph/make-node 1 nil)))
|
||
#+end_src
|
||
|
||
#+begin_src dot :noweb yes :file git-graphs-no-auto-grouping.svg :exports results
|
||
<<graph-no-auto-grouping()>>
|
||
#+end_src
|
||
|
||
#+RESULTS[91bedd3cab2a02d3083d10217462e07aa8eb0be0]:
|
||
[[file:static/ox-hugo/git-graphs-no-auto-grouping.svg]]
|
||
|
||
*** Graph with automatic grouping
|
||
|
||
#+name: graph-with-auto-grouping
|
||
#+begin_src emacs-lisp :exports code :results silent
|
||
(git-graph/to-graphviz-pretty
|
||
"autogroups"
|
||
(git-graph/group-topo
|
||
(list (git-graph/make-node 5 '(4) '((label . master)))
|
||
(git-graph/make-node 4 '(3 7))
|
||
(git-graph/make-node 3 '(2))
|
||
(git-graph/make-node 7 '(6) '((label . develop)))
|
||
(git-graph/make-node 6 '(2))
|
||
(git-graph/make-node 2 '(1))
|
||
(git-graph/make-node 1 nil))))
|
||
#+end_src
|
||
|
||
#+begin_src dot :noweb yes :file git-graphs-with-auto-grouping.svg :exports results
|
||
<<graph-with-auto-grouping()>>
|
||
#+end_src
|
||
|
||
#+RESULTS[fa116b45cd590ae9cb00517bb3ed51dbab357592]:
|
||
[[file:static/ox-hugo/git-graphs-with-auto-grouping.svg]]
|
||
|
||
** Graphing a Git Repository
|
||
|
||
Satisfied that I had all the necessary tools to start graphing real
|
||
git repositories, I created an example repository to test against.
|
||
|
||
*** Creating a Sample Repository
|
||
|
||
Using the following script, I created a sample repository to test
|
||
against. I performed the following actions:
|
||
|
||
- Forked a develop branch from master.
|
||
- Forked a feature branch from develop, with two commits.
|
||
- Added another commit to develop.
|
||
- Forked a second feature branch from develop, with two commits.
|
||
- Merged the second feature branch to develop.
|
||
- Merged develop to master and tagged it.
|
||
|
||
#+begin_src sh :exports results :results silent
|
||
rm -rf /tmp/test.git
|
||
#+end_src
|
||
#+begin_src sh :exports both :results silent
|
||
mkdir /tmp/test.git
|
||
cd /tmp/test.git
|
||
git init
|
||
touch README
|
||
git add README
|
||
git commit -m 'initial'
|
||
git commit --allow-empty -m 'first'
|
||
git checkout -b develop
|
||
git commit --allow-empty -m 'second'
|
||
git checkout -b feature-1
|
||
git commit --allow-empty -m 'feature 1'
|
||
git commit --allow-empty -m 'feature 1 again'
|
||
git checkout develop
|
||
git commit --allow-empty -m 'third'
|
||
git checkout -b feature-2
|
||
git commit --allow-empty -m 'feature 2'
|
||
git commit --allow-empty -m 'feature 2 again'
|
||
git checkout develop
|
||
git merge --no-ff feature-2
|
||
git checkout master
|
||
git merge --no-ff develop
|
||
git tag -a 1.0 -m '1.0!'
|
||
#+end_src
|
||
*** Generating a Graph From a Git Branch
|
||
|
||
The first order of business was to have a way to call out to git and
|
||
return the results:
|
||
|
||
#+name: git-graph/from-git
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/git-execute (repo-url command &rest args)
|
||
(with-temp-buffer
|
||
(shell-command (format "git -C \"%s\" %s"
|
||
repo-url
|
||
(string-join (cons command args)
|
||
" "))
|
||
t)
|
||
(buffer-string)))
|
||
#+end_src
|
||
|
||
Next, I needed to get the list of commits for a branch in topological
|
||
order, with a list of parent commits for each. It turns out git
|
||
provides exactly that via its =rev-list= command.
|
||
|
||
#+name: git-graph/from-git
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/git-rev-list (repo-url head)
|
||
(-map (lambda (line) (split-string line))
|
||
(split-string (git-graph/git-execute
|
||
repo-url
|
||
"rev-list" "--topo-order" "--parents" head)
|
||
"\n" t)))
|
||
#+end_src
|
||
|
||
I also wanted to label branch heads wherever possible. To do this, I
|
||
looked up the revision name from git, discarding it if it was relative
|
||
to some other named commit.
|
||
|
||
#+name: git-graph/from-git
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/git-label (repo-url rev)
|
||
(let ((name (string-trim
|
||
(git-graph/git-execute repo-url
|
||
"name-rev" "--name-only" rev))))
|
||
(unless (s-contains? "~" name)
|
||
name)))
|
||
#+end_src
|
||
|
||
Generating the graph for a single branch was as simple as iterating
|
||
over each commit and creating a node for it.
|
||
|
||
#+name: git-graph/from-git
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/git-graphs-head (repo-url head)
|
||
(git-graph/group-topo
|
||
(-map (lambda (rev-with-parents)
|
||
(let* ((rev (car rev-with-parents))
|
||
(parents (cdr rev-with-parents))
|
||
(label (git-graph/git-label repo-url rev)))
|
||
(git-graph/make-node rev parents
|
||
`((label . ,label)))))
|
||
(git-graph/git-rev-list repo-url head))))
|
||
#+end_src
|
||
|
||
Here's the result of graphing the =master= branch:
|
||
|
||
#+name: graph-git-branch
|
||
#+begin_src emacs-lisp
|
||
(git-graph/to-graphviz-pretty
|
||
"git"
|
||
(git-graph/git-graphs-head
|
||
"/tmp/test.git"
|
||
"master"))
|
||
#+end_src
|
||
|
||
#+begin_src dot :file git-graphs-branch.svg :noweb yes
|
||
<<graph-git-branch()>>
|
||
#+end_src
|
||
|
||
#+RESULTS[e971f68020b770b27fa6d08eaaec85798e8da4a2]:
|
||
[[file:static/ox-hugo/git-graphs-branch.svg]]
|
||
|
||
*** Graphing Multiple Branches
|
||
|
||
To graph multiple branches, I needed a function for combining
|
||
histories. To do so, I simply append any nodes I don't already know
|
||
about in the first history from the second.
|
||
|
||
#+name: git-graph/adder
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/+ (a b)
|
||
(append a
|
||
(-remove (lambda (node)
|
||
(assoc (git-graph/node-id node) a))
|
||
b)))
|
||
#+end_src
|
||
|
||
From there, all that remained was to accumulate the branch histories
|
||
and output the complete graph:
|
||
|
||
#+name: git-graph/from-git
|
||
#+begin_src emacs-lisp
|
||
(defun git-graph/git-load (repo-url heads)
|
||
(-reduce #'git-graph/+
|
||
(-map (lambda (head)
|
||
(git-graph/git-graphs-head repo-url head))
|
||
heads)))
|
||
#+end_src
|
||
|
||
And here's the example repository, graphed in full:
|
||
|
||
#+name: graph-git-repo
|
||
#+begin_src emacs-lisp
|
||
(git-graph/to-graphviz-pretty
|
||
"git"
|
||
(git-graph/git-load
|
||
"/tmp/test.git"
|
||
'("master" "feature-1")))
|
||
#+end_src
|
||
|
||
#+begin_src dot :file git-graphs-repo.svg :noweb yes
|
||
<<graph-git-repo()>>
|
||
#+end_src
|
||
|
||
#+RESULTS[0d4e90afa31090ce57eeb60b7f40c0579e3fbc1e]:
|
||
[[file:static/ox-hugo/git-graphs-repo.svg]]
|
||
|
||
** Things I may add in the future
|
||
*** Limiting Commits to Graph
|
||
|
||
Running this against repos with any substantial history can make the
|
||
graph unwieldy. It'd be a good idea to abstract out the commit list
|
||
fetching, and modify it to support different ways of limiting the
|
||
history to display.
|
||
|
||
Ideas would include:
|
||
- Specifying commit ranges
|
||
- Stopping at a common ancestor to all graphed branches (e.g., using
|
||
=git-merge-base=).
|
||
- Other git commit limiting options, like searches, showing only merge
|
||
or non-merge commits, etc.
|
||
|
||
*** Collapsing History
|
||
|
||
Another means of reducing the size of the resulting graph would be to
|
||
collapse unimportant sections of it. It should be possible to collapse
|
||
a section of the graph, showing a count of skipped nodes.
|
||
|
||
The difficult part would be determining what parts aren't worth
|
||
drawing. Something like this would be handy, though, for concisely
|
||
graphing the state of multiple ongoing development branches (say, to
|
||
get a picture of what's been going on since the last release, and
|
||
what's still incomplete).
|
||
|
||
#+begin_src dot :file git-graphs-long.svg
|
||
digraph G {
|
||
rankdir="LR";
|
||
bgcolor="transparent";
|
||
node[width=0.15,height=0.15,shape=point,color=white];
|
||
edge[weight=2,arrowhead=none,color=white];
|
||
node[group=main];
|
||
1 -> 2 -> 3 -> 4 -> 5;
|
||
node[group=branch];
|
||
2 -> 6 -> 7 -> 8 -> 9 -> 10 -> 4;
|
||
}
|
||
#+end_src
|
||
|
||
#+caption: A graph with multiple nodes on a branch.
|
||
#+RESULTS[6d6237fcc49d1bbc21685b447d7065ba1faf907e]:
|
||
[[file:static/ox-hugo/git-graphs-long.svg]]
|
||
|
||
|
||
#+begin_src dot :file git-graphs-collapsed.svg
|
||
digraph G {
|
||
rankdir="LR";
|
||
bgcolor="transparent";
|
||
node[width=0.15,height=0.15,shape=point,color=white];
|
||
edge[weight=2,arrowhead=none,color=white,fontcolor=white];
|
||
node[group=main];
|
||
1 -> 2 -> 3 -> 4 -> 5;
|
||
node[group=branch];
|
||
2 -> 6;
|
||
6 -> 10[style=dashed,label="+3"];
|
||
10 -> 4;
|
||
}
|
||
#+end_src
|
||
|
||
#+caption: The same graph, collapsed.
|
||
#+RESULTS[4bf40f7b350a8d92ddc70098eb48d8a0d50f432b]:
|
||
[[file:static/ox-hugo/git-graphs-collapsed.svg]]
|
||
|
||
|
||
*** Clean up and optimize the code a bit
|
||
|
||
Some parts of this (particularly, the grouping) are probably pretty
|
||
inefficient. If this turns out to actually be useful, I may take
|
||
another crack at it.
|
||
|
||
** Final Code
|
||
|
||
In case anyone would like to use this code for anything, or maybe just
|
||
pick it apart and play around with it, all the Emacs Lisp code in this
|
||
post is collected into a single file below:
|
||
|
||
#+begin_src emacs-lisp :noweb yes :exports code :tangle "static/files/git-graph.el"
|
||
;;; git-graph.el --- Generate git-style graphs using graphviz
|
||
|
||
;; Copyright (c) 2015 Correl Roush <correl@gmail.com>
|
||
|
||
;;; License:
|
||
|
||
;; This program is free software; you can redistribute it and/or modify
|
||
;; it under the terms of the GNU General Public License as published by
|
||
;; the Free Software Foundation; either version 3, or (at your option)
|
||
;; any later version.
|
||
;;
|
||
;; This program is distributed in the hope that it will be useful,
|
||
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
;; GNU General Public License for more details.
|
||
;;
|
||
;; You should have received a copy of the GNU General Public License
|
||
;; along with GNU Emacs; see the file COPYING. If not, write to the
|
||
;; Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
||
;; Boston, MA 02110-1301, USA.
|
||
|
||
;;; Commentary:
|
||
|
||
;;; Code:
|
||
|
||
(require 'dash)
|
||
|
||
<<git-graph/structure>>
|
||
|
||
<<git-graph/adder>>
|
||
|
||
<<git-graph/to-graphviz>>
|
||
|
||
<<git-graph/to-graphviz-nodes>>
|
||
|
||
<<git-graph/to-graphviz-edges>>
|
||
|
||
<<git-graph/group-topo>>
|
||
|
||
<<git-graph/from-git>>
|
||
|
||
(provide 'git-graph)
|
||
;;; git-graph.el ends here
|
||
#+end_src
|
||
|
||
Download: [[file:/files/git-graph.el][git-graph.el]]
|
||
* DONE Use a different theme when publishing Org files :emacs:org_mode:
|
||
CLOSED: [2016-02-23 Tue]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: org-publish-with-theme
|
||
:EXPORT_HUGO_SLUG: org-publish-with-theme
|
||
:END:
|
||
#+KEYWORDS: emacs org-mode themes
|
||
|
||
I've been using [[https://github.com/cpaulik/emacs-material-theme][material-theme]] lately, and I sometimes switch around,
|
||
but I've found that [[https://github.com/bbatsov/solarized-emacs][solarized]] produces the best exported code block
|
||
results. To avoid having to remember to switch themes when exporting,
|
||
I wrote a quick wrapper for org-export to do it for me:
|
||
|
||
#+BEGIN_SRC emacs-lisp :exports code
|
||
(defun my/with-theme (theme fn &rest args)
|
||
(let ((current-themes custom-enabled-themes))
|
||
(mapcar #'disable-theme custom-enabled-themes)
|
||
(load-theme theme t)
|
||
(let ((result (apply fn args)))
|
||
(mapcar #'disable-theme custom-enabled-themes)
|
||
(mapcar (lambda (theme) (load-theme theme t)) current-themes)
|
||
result)))
|
||
|
||
(advice-add #'org-export-to-file :around (apply-partially #'my/with-theme 'solarized-dark))
|
||
(advice-add #'org-export-to-buffer :around (apply-partially #'my/with-theme 'solarized-dark))
|
||
#+END_SRC
|
||
|
||
Voilà, no more bizarrely formatted code block exports from whatever
|
||
theme I might have loaded at the time :)
|
||
* DONE Recursive HTTP Requests with Elm :programming:elm:
|
||
CLOSED: [2018-01-22 Mon]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: recursive-http-with-elm
|
||
:EXPORT_HUGO_SLUG: recursive-http-with-elm
|
||
:END:
|
||
|
||
So I got the idea in my head that I wanted to pull data from the
|
||
GitLab / GitHub APIs in my Elm app. This seemed straightforward
|
||
enough; just wire up an HTTP request and a JSON decoder, and off I go.
|
||
Then I remember, oh crap... like any sensible API with a potentially
|
||
huge amount of data behind it, the results come back /paginated/. For
|
||
anyone unfamiliar, this means that a single API request for a list of,
|
||
say, repositories, is only going to return up to some maximum number
|
||
of results. If there are more results available, there will be a
|
||
reference to additional /pages/ of results, that you can then fetch
|
||
with /another/ API request. My single request decoding only the
|
||
results returned /from/ that single request wasn't going to cut it.
|
||
|
||
I had a handful of problems to solve. I needed to:
|
||
|
||
- Detect when additional results were available.
|
||
- Parse out the URL to use to fetch the next page of results.
|
||
- Continue fetching results until none remained.
|
||
- Combine all of the results, maintaining their order.
|
||
|
||
** Are there more results?
|
||
|
||
The first two bullet points can be dealt with by parsing and
|
||
inspecting the response header. Both GitHub and GitLab embed
|
||
pagination links in the [[https://www.w3.org/wiki/LinkHeader][HTTP Link header]]. As I'm interested in
|
||
consuming pages until no further results remain, I'll be looking for a
|
||
link in the header with the relationship "next". If I find one, I know
|
||
I need to hit the associated URL to fetch more results. If I don't
|
||
find one, I'm done!
|
||
|
||
#+CAPTION: Example GitHub Link header
|
||
#+BEGIN_SRC http
|
||
Link: <https://api.github.com/user/repos?page=3&per_page=100>; rel="next",
|
||
<https://api.github.com/user/repos?page=50&per_page=100>; rel="last"
|
||
#+END_SRC
|
||
|
||
Parsing this stuff out went straight into a utility module.
|
||
|
||
#+BEGIN_SRC elm
|
||
module Paginated.Util exposing (links)
|
||
|
||
import Dict exposing (Dict)
|
||
import Maybe.Extra
|
||
import Regex
|
||
|
||
|
||
{-| Parse an HTTP Link header into a dictionary. For example, to look
|
||
for a link to additional results in an API response, you could do the
|
||
following:
|
||
|
||
Dict.get "Link" response.headers
|
||
|> Maybe.map links
|
||
|> Maybe.andThen (Dict.get "next")
|
||
|
||
-}
|
||
links : String -> Dict String String
|
||
links s =
|
||
let
|
||
toTuples xs =
|
||
case xs of
|
||
[ Just a, Just b ] ->
|
||
Just ( b, a )
|
||
|
||
_ ->
|
||
Nothing
|
||
in
|
||
Regex.find
|
||
Regex.All
|
||
(Regex.regex "<(.*?)>; rel=\"(.*?)\"")
|
||
s
|
||
|> List.map .submatches
|
||
|> List.map toTuples
|
||
|> Maybe.Extra.values
|
||
|> Dict.fromList
|
||
#+END_SRC
|
||
|
||
A little bit of regular expression magic, tuples, and
|
||
=Maybe.Extra.values= to keep the matches, and now I've got my
|
||
(=Maybe=) URL.
|
||
|
||
** Time to make some requests
|
||
|
||
Now's the time to define some types. I'll need a =Request=, which will
|
||
be similar to a standard =Http.Request=, with a /slight/ difference.
|
||
|
||
#+BEGIN_SRC elm
|
||
type alias RequestOptions a =
|
||
{ method : String
|
||
, headers : List Http.Header
|
||
, url : String
|
||
, body : Http.Body
|
||
, decoder : Decoder a
|
||
, timeout : Maybe Time.Time
|
||
, withCredentials : Bool
|
||
}
|
||
|
||
|
||
type Request a
|
||
= Request (RequestOptions a)
|
||
#+END_SRC
|
||
|
||
What separates it from a basic =Http.Request= is the =decoder= field
|
||
instead of an =expect= field. The =expect= field in an HTTP request is
|
||
responsible for parsing the full response into whatever result the
|
||
caller wants. For my purposes, I always intend to be hitting a JSON
|
||
API returning a list of items, and I have my own designs on parsing
|
||
bits of the request to pluck out the headers. Therefore, I expose only
|
||
a slot for including a JSON decoder representing the type of item I'll
|
||
be getting a collection of.
|
||
|
||
I'll also need a =Response=, which will either be =Partial=
|
||
(containing the results from the response, plus a =Request= for
|
||
getting the next batch), or =Complete=.
|
||
|
||
#+BEGIN_SRC elm
|
||
type Response a
|
||
= Partial (Request a) (List a)
|
||
| Complete (List a)
|
||
#+END_SRC
|
||
|
||
Sending the request isn't too bad. I can just convert my request into
|
||
an =Http.Request=, and use =Http.send=.
|
||
|
||
#+BEGIN_SRC elm
|
||
send :
|
||
(Result Http.Error (Response a) -> msg)
|
||
-> Request a
|
||
-> Cmd msg
|
||
send resultToMessage request =
|
||
Http.send resultToMessage <|
|
||
httpRequest request
|
||
|
||
|
||
httpRequest : Request a -> Http.Request (Response a)
|
||
httpRequest (Request options) =
|
||
Http.request
|
||
{ method = options.method
|
||
, headers = options.headers
|
||
, url = options.url
|
||
, body = options.body
|
||
, expect = expect options
|
||
, timeout = options.timeout
|
||
, withCredentials = options.withCredentials
|
||
}
|
||
|
||
|
||
expect : RequestOptions a -> Http.Expect (Response a)
|
||
expect options =
|
||
Http.expectStringResponse (fromResponse options)
|
||
#+END_SRC
|
||
|
||
All of my special logic for handling the headers, mapping the decoder
|
||
over the results, and packing them up into a =Response= is baked into
|
||
my =Http.Request= via a private =fromResponse= translator:
|
||
|
||
#+BEGIN_SRC elm
|
||
fromResponse :
|
||
RequestOptions a
|
||
-> Http.Response String
|
||
-> Result String (Response a)
|
||
fromResponse options response =
|
||
let
|
||
items : Result String (List a)
|
||
items =
|
||
Json.Decode.decodeString
|
||
(Json.Decode.list options.decoder)
|
||
response.body
|
||
|
||
nextPage =
|
||
Dict.get "Link" response.headers
|
||
|> Maybe.map Paginated.Util.links
|
||
|> Maybe.andThen (Dict.get "next")
|
||
in
|
||
case nextPage of
|
||
Nothing ->
|
||
Result.map Complete items
|
||
|
||
Just url ->
|
||
Result.map
|
||
(Partial (request { options | url = url }))
|
||
items
|
||
#+END_SRC
|
||
|
||
** Putting it together
|
||
|
||
Now, I can make my API request, and get back a response with
|
||
potentially partial results. All that needs to be done now is to make
|
||
my request, and iterate on the results I get back in my =update=
|
||
method.
|
||
|
||
To make things a bit easier, I add a method for concatenating two
|
||
responses:
|
||
|
||
#+BEGIN_SRC elm
|
||
update : Response a -> Response a -> Response a
|
||
update old new =
|
||
case ( old, new ) of
|
||
( Complete items, _ ) ->
|
||
Complete items
|
||
|
||
( Partial _ oldItems, Complete newItems ) ->
|
||
Complete (oldItems ++ newItems)
|
||
|
||
( Partial _ oldItems, Partial request newItems ) ->
|
||
Partial request (oldItems ++ newItems)
|
||
#+END_SRC
|
||
|
||
Putting it all together, I get a fully functional test app that
|
||
fetches a paginated list of repositories from GitLab, and renders them
|
||
when I've fetched them all:
|
||
|
||
#+BEGIN_SRC elm
|
||
module Example exposing (..)
|
||
|
||
import Html exposing (Html)
|
||
import Http
|
||
import Json.Decode exposing (field, string)
|
||
import Paginated exposing (Response(..))
|
||
|
||
|
||
type alias Model =
|
||
{ repositories : Maybe (Response String) }
|
||
|
||
|
||
type Msg
|
||
= GotRepositories (Result Http.Error (Paginated.Response String))
|
||
|
||
|
||
main : Program Never Model Msg
|
||
main =
|
||
Html.program
|
||
{ init = init
|
||
, update = update
|
||
, view = view
|
||
, subscriptions = \_ -> Sub.none
|
||
}
|
||
|
||
|
||
init : ( Model, Cmd Msg )
|
||
init =
|
||
( { repositories = Nothing }
|
||
, getRepositories
|
||
)
|
||
|
||
|
||
update : Msg -> Model -> ( Model, Cmd Msg )
|
||
update msg model =
|
||
case msg of
|
||
GotRepositories (Ok response) ->
|
||
( { model
|
||
| repositories =
|
||
case model.repositories of
|
||
Nothing ->
|
||
Just response
|
||
|
||
Just previous ->
|
||
Just (Paginated.update previous response)
|
||
}
|
||
, case response of
|
||
Partial request _ ->
|
||
Paginated.send GotRepositories request
|
||
|
||
Complete _ ->
|
||
Cmd.none
|
||
)
|
||
|
||
GotRepositories (Err _) ->
|
||
( { model | repositories = Nothing }
|
||
, Cmd.none
|
||
)
|
||
|
||
|
||
view : Model -> Html Msg
|
||
view model =
|
||
case model.repositories of
|
||
Nothing ->
|
||
Html.div [] [ Html.text "Loading" ]
|
||
|
||
Just (Partial _ _) ->
|
||
Html.div [] [ Html.text "Loading..." ]
|
||
|
||
Just (Complete repos) ->
|
||
Html.ul [] <|
|
||
List.map
|
||
(\x -> Html.li [] [ Html.text x ])
|
||
repos
|
||
|
||
|
||
getRepositories : Cmd Msg
|
||
getRepositories =
|
||
Paginated.send GotRepositories <|
|
||
Paginated.get
|
||
"http://git.phoenixinquis.net/api/v4/projects?per_page=5"
|
||
(field "name" string)
|
||
#+END_SRC
|
||
|
||
** There's got to be a better way
|
||
|
||
I've got it working, and it's working well. However, it's kind of a
|
||
pain to use. It's nice that I can play with the results as they come
|
||
in by peeking into the =Partial= structure, but it's a real chore to
|
||
have to stitch the results together in my application's =update=
|
||
method. It'd be nice if I could somehow encapsulate that behavior in
|
||
my request and not have to worry about the pagination at all in my
|
||
app.
|
||
|
||
It just so happens that, with Tasks, I can.
|
||
|
||
/Feel free to check out the full library documentation and code
|
||
referenced in this post [[http://package.elm-lang.org/packages/correl/elm-paginated/1.0.1][here]]./
|
||
|
||
/Continue on with part two, [[relref:cleaner-recursive-http-with-elm-tasks][Cleaner Recursive HTTP Requests with Elm
|
||
Tasks]]./
|
||
|
||
* DONE Cleaner Recursive HTTP Requests with Elm Tasks :programming:elm:
|
||
CLOSED: [2018-01-23 Tue]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: cleaner-recursive-http-with-elm-tasks
|
||
:EXPORT_HUGO_SLUG: cleaner-recursive-http-with-elm-tasks
|
||
:END:
|
||
/Continued from part one, [[relref:recursive-http-with-elm][Recursive HTTP Requests with Elm]]./
|
||
|
||
In [[relref:recursive-http-with-elm][my last post]], I described my first pass at building a library to
|
||
fetch data from a paginated JSON REST API. It worked, but it wasn't
|
||
too clean. In particular, the handling of the multiple pages and
|
||
concatenation of results was left up to the calling code. Ideally,
|
||
both of these concerns should be handled by the library, letting the
|
||
application focus on working with a full result set. Using Elm's
|
||
Tasks, we can achieve exactly that!
|
||
|
||
** What's a Task?
|
||
|
||
A [[http://package.elm-lang.org/packages/elm-lang/core/5.1.1/Task][Task]] is a data structure in Elm which represents an asynchronous
|
||
operation that may fail, which can be mapped and *chained*. What this
|
||
means is, we can create an action, transform it, and chain it with
|
||
additional actions, building up a complex series of things to do into
|
||
a single =Task=, which we can then package up into a [[http://package.elm-lang.org/packages/elm-lang/core/5.1.1/Platform-Cmd#Cmd][Cmd]] and hand to
|
||
the Elm runtime to perform. You can think of it like building up a
|
||
[[https://en.wikipedia.org/wiki/Futures_and_promises][Future or Promise]], setting up a sort of [[https://en.wikipedia.org/wiki/Callback_(computer_programming)][callback]] chain of mutations
|
||
and follow-up actions to be taken. The Elm runtime will work its way
|
||
through the chain and hand your application back the result in the
|
||
form of a =Msg=.
|
||
|
||
So, tasks sound great!
|
||
|
||
** Moving to Tasks
|
||
|
||
Just to get things rolling, let's quit using =Http.send=, and instead
|
||
prepare a simple =toTask= function leveraging the very handy
|
||
=Http.toTask=. This'll give us a place to start building up some more
|
||
complex behavior.
|
||
|
||
#+BEGIN_SRC elm
|
||
send :
|
||
(Result Http.Error (Response a) -> msg)
|
||
-> Request a
|
||
-> Cmd msg
|
||
send resultToMessage request =
|
||
toTask request
|
||
|> Task.attempt resultToMessage
|
||
|
||
|
||
toTask : Request a -> Task Http.Error (Response a)
|
||
toTask =
|
||
httpRequest >> Http.toTask
|
||
#+END_SRC
|
||
|
||
** Shifting the recursion
|
||
|
||
Now, for the fun bit. We want, when a request completes, to inspect
|
||
the result. If the task failed, we do nothing. If it succeeded, we
|
||
move on to checking the response. If we have a =Complete= response,
|
||
we're done. If we do not, we want to build another task for the next
|
||
request, and start a new iteration on that.
|
||
|
||
All that needs to be done here is to chain our response handling using
|
||
=Task.andThen=, and either recurse to continue the chain with the next
|
||
=Task=, or wrap up the final results with =Task.succeed=!
|
||
|
||
#+BEGIN_SRC elm
|
||
recurse :
|
||
Task Http.Error (Response a)
|
||
-> Task Http.Error (Response a)
|
||
recurse =
|
||
Task.andThen
|
||
(\response ->
|
||
case response of
|
||
Partial request _ ->
|
||
httpRequest request
|
||
|> Http.toTask
|
||
|> recurse
|
||
|
||
Complete _ ->
|
||
Task.succeed response
|
||
)
|
||
#+END_SRC
|
||
|
||
That wasn't so bad. The function recursion almost seems like cheating:
|
||
I'm able to build up a whole chain of requests /based/ on the results
|
||
without actually /having/ the results yet! The =Task= lets us define a
|
||
complete plan for what to do with the results, using what we know
|
||
about the data structures flowing through to make decisions and tack
|
||
on additional things to do.
|
||
|
||
** Accumulating results
|
||
|
||
There's just one thing left to do: we're not accumulating results yet.
|
||
We're just handing off the results of the final request, which isn't
|
||
too helpful to the caller. We're also still returning our Response
|
||
structure, which is no longer necessary, since we're not bothering
|
||
with returning incomplete requests anymore.
|
||
|
||
Cleaning up the types is pretty easy. It's just a matter of switching
|
||
out some instances of =Response a= with =List a= in our type
|
||
declarations...
|
||
|
||
#+BEGIN_SRC elm
|
||
send :
|
||
(Result Http.Error (List a) -> msg)
|
||
-> Request a
|
||
-> Cmd msg
|
||
|
||
|
||
toTask : Request a -> Task Http.Error (List a)
|
||
|
||
|
||
recurse :
|
||
Task Http.Error (Response a)
|
||
-> Task Http.Error (List a)
|
||
#+END_SRC
|
||
|
||
|
||
...then changing our =Complete= case to return the actual items:
|
||
|
||
#+BEGIN_SRC elm
|
||
Complete xs ->
|
||
Task.succeed xs
|
||
#+END_SRC
|
||
|
||
The final step, then, is to accumulate the results. Turns out this is
|
||
*super* easy. We already have an =update= function that combines two
|
||
responses, so we can map /that/ over our next request task so that it
|
||
incorporates the previous request's results!
|
||
|
||
#+BEGIN_SRC elm
|
||
Partial request _ ->
|
||
httpRequest request
|
||
|> Http.toTask
|
||
|> Task.map (update response)
|
||
|> recurse
|
||
#+END_SRC
|
||
|
||
** Tidying up
|
||
|
||
Things are tied up pretty neatly, now! Calling code no longer needs to
|
||
care whether the JSON endpoints its calling paginate their results,
|
||
they'll receive everything they asked for as though it were a single
|
||
request. Implementation details like the =Response= structure,
|
||
=update= method, and =httpRequest= no longer need to be exposed.
|
||
=toTask= can be exposed now as a convenience to anyone who wants to
|
||
perform further chaining on their calls.
|
||
|
||
Now that there's a cleaner interface to the module, the example app is
|
||
looking a lot cleaner now, too:
|
||
|
||
#+BEGIN_SRC elm
|
||
module Example exposing (..)
|
||
|
||
import Html exposing (Html)
|
||
import Http
|
||
import Json.Decode exposing (field, string)
|
||
import Paginated
|
||
|
||
|
||
type alias Model =
|
||
{ repositories : Maybe (List String) }
|
||
|
||
|
||
type Msg
|
||
= GotRepositories (Result Http.Error (List String))
|
||
|
||
|
||
main : Program Never Model Msg
|
||
main =
|
||
Html.program
|
||
{ init = init
|
||
, update = update
|
||
, view = view
|
||
, subscriptions = \_ -> Sub.none
|
||
}
|
||
|
||
|
||
init : ( Model, Cmd Msg )
|
||
init =
|
||
( { repositories = Nothing }
|
||
, getRepositories
|
||
)
|
||
|
||
|
||
update : Msg -> Model -> ( Model, Cmd Msg )
|
||
update msg model =
|
||
case msg of
|
||
GotRepositories result ->
|
||
( { model | repositories = Result.toMaybe result }
|
||
, Cmd.none
|
||
)
|
||
|
||
|
||
view : Model -> Html Msg
|
||
view model =
|
||
case model.repositories of
|
||
Nothing ->
|
||
Html.div [] [ Html.text "Loading" ]
|
||
|
||
Just repos ->
|
||
Html.ul [] <|
|
||
List.map
|
||
(\x -> Html.li [] [ Html.text x ])
|
||
repos
|
||
|
||
|
||
getRepositories : Cmd Msg
|
||
getRepositories =
|
||
Paginated.send GotRepositories <|
|
||
Paginated.get
|
||
"http://git.phoenixinquis.net/api/v4/projects?per_page=5"
|
||
(field "name" string)
|
||
#+END_SRC
|
||
|
||
So, there we have it! Feel free to check out the my complete
|
||
=Paginated= library on the [[http://package.elm-lang.org/packages/correl/elm-paginated/latest][Elm package index]], or on [[https://github.com/correl/elm-paginated][GitHub]]. Hopefully
|
||
you'll find it or this post useful. I'm still finding my way around
|
||
Elm, so any and all feedback is quite welcome :)
|
||
* DONE How Does The Phillips Hue Wake-Up Feature Work? :home_automation:
|
||
CLOSED: [2018-03-13 Tue]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: hue-wake-up
|
||
:EXPORT_HUGO_SLUG: hue-wake-up
|
||
:header-args:http: :post anonymize(json=*this*) :resolve bridge:80:192.168.1.199 :var username="uCpPPhYlWIJdOvGuyrsM2EtmaeiVvcTX0WEdAn0P" :pretty :exports both :wrap SRC js
|
||
:END:
|
||
#+name: anonymize
|
||
#+BEGIN_SRC emacs-lisp :var json="" :exports none :results silent
|
||
;; Anonymize user/manufacturer IDs
|
||
(->>
|
||
json
|
||
(s-replace "oV5vUaXuBwEAA6sjnvqr8n6fBLlzWLjG4x4SIyD8"
|
||
"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx")
|
||
(s-replace "oV5vUaXuBwEAA6sjnvqr8n6fBLlzWLjG"
|
||
"xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"))
|
||
#+END_SRC
|
||
|
||
I recently got myself a set of Phillips Hue White and Color Ambiance
|
||
lights. One of the features I was looking forward to in particular
|
||
(besides playing with all the color options) was setting a wake-up
|
||
alarm with the lights gradually brightening. This was pretty painless
|
||
to get set up using the phone app. I'm pretty happy with the result,
|
||
but there's certainly some things I wouldn't mind tweaking. For
|
||
example, the initial brightness of the bulbs (at the lowest setting)
|
||
still seems a bit bright, so I might want to delay the bedside lamps
|
||
and let the more distant lamp start fading in first. I also want to
|
||
see if I can fiddle it into transitioning between some colors to get
|
||
more of a sunrise effect (perhaps "rising" from the other side of the
|
||
room, with the light spreading towards the head of the bed).
|
||
|
||
Figuring out how the wake-up settings that the app installed on my
|
||
bridge seemed a good first step towards introducing my own
|
||
customizations.
|
||
|
||
Information on getting access to a Hue bridge to make REST API calls
|
||
to it can be found in the [[https://www.developers.meethue.com/documentation/getting-started][Hue API getting started guide]].
|
||
|
||
** My wake-up settings
|
||
My wake-up is scheduled for 7:00 to gradually brighten the lights with
|
||
a half-hour fade-in each weekday. I also toggled on the setting to
|
||
automatically turn the lights off at 9:00.
|
||
|
||
#+BEGIN_CENTER
|
||
[[file:static/images/Screenshot_20180313-182434.png]] [[file:static/images/Screenshot_20180313-182438.png]]
|
||
#+END_CENTER
|
||
|
||
** Finding things on the bridge
|
||
|
||
The most natural starting point is to check the schedules. Right off
|
||
the bat, I find what I'm after:
|
||
|
||
*** The schedule ...
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/schedules/1
|
||
#+END_SRC
|
||
|
||
#+RESULTS[185c8bd9c6da61034cb699944dfb1827d2d08282]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"name": "Wake up",
|
||
"description": "L_04_fidlv_start wake up",
|
||
"command": {
|
||
"address": "/api/xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx/sensors/2/state",
|
||
"body": {
|
||
"flag": true
|
||
},
|
||
"method": "PUT"
|
||
},
|
||
"localtime": "W124/T06:30:00",
|
||
"time": "W124/T10:30:00",
|
||
"created": "2018-03-11T19:46:54",
|
||
"status": "enabled",
|
||
"recycle": true
|
||
}
|
||
#+END_SRC
|
||
|
||
This is a recurring schedule item that runs every weekday at 6:30. We
|
||
can tell this by looking at the =localtime= field. From the
|
||
documentation on [[https://www.developers.meethue.com/documentation/datatypes-and-time-patterns#16_time_patterns][time patterns]], we can see that it's a recurring time
|
||
pattern specifying days of the week as a bitmask, and a time (6:30).
|
||
|
||
#+CAPTION: Unraveling the weekday portion
|
||
| <l> |
|
||
| =0MTWTFSS= |
|
||
| =01111100= (124 in decimal) |
|
||
|
||
Since this schedule is enabled, we can be assured that it will run,
|
||
and in doing so, will issue a =PUT= to a sensors endpoint, setting a
|
||
flag to true.
|
||
|
||
*** ... triggers the sensor ...
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/sensors/2
|
||
#+END_SRC
|
||
|
||
#+RESULTS[6660e5539302c8863c2d7cd9417e3996c35b2b87]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"state": {
|
||
"flag": false,
|
||
"lastupdated": "2018-03-13T13:00:00"
|
||
},
|
||
"config": {
|
||
"on": true,
|
||
"reachable": true
|
||
},
|
||
"name": "Sensor for wakeup",
|
||
"type": "CLIPGenericFlag",
|
||
"modelid": "WAKEUP",
|
||
"manufacturername": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
||
"swversion": "A_1801260942",
|
||
"uniqueid": "L_04_fidlv",
|
||
"recycle": true
|
||
}
|
||
#+END_SRC
|
||
|
||
The sensor is what's /really/ setting things in motion. Here we've got
|
||
a [[https://www.developers.meethue.com/documentation/supported-sensors#clipSensors][generic CLIP flag sensor]] that is triggered exclusively by our
|
||
schedule. Essentially, by updating the flag state, we trigger the
|
||
sensor.
|
||
|
||
*** ... triggers a rule ...
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/rules/1
|
||
#+END_SRC
|
||
|
||
#+RESULTS[449d90a31c32d4630701c75e4655ccb6378d8655]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"name": "L_04_fidlv_Start",
|
||
"owner": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
||
"created": "2018-03-11T19:46:51",
|
||
"lasttriggered": "2018-03-13T10:30:00",
|
||
"timestriggered": 2,
|
||
"status": "enabled",
|
||
"recycle": true,
|
||
"conditions": [
|
||
{
|
||
"address": "/sensors/2/state/flag",
|
||
"operator": "eq",
|
||
"value": "true"
|
||
}
|
||
],
|
||
"actions": [
|
||
{
|
||
"address": "/groups/1/action",
|
||
"method": "PUT",
|
||
"body": {
|
||
"scene": "7GJer2-5ahGIqz6"
|
||
}
|
||
},
|
||
{
|
||
"address": "/schedules/2",
|
||
"method": "PUT",
|
||
"body": {
|
||
"status": "enabled"
|
||
}
|
||
}
|
||
]
|
||
}
|
||
#+END_SRC
|
||
|
||
Now things are happening. Looking at the conditions, we can see that
|
||
this rule triggers when the wakeup sensor updates, and its flag is set
|
||
to =true=. When that happens, the bridge will iterate through its
|
||
rules, find that the above condition has been met, and iterate through
|
||
each of the actions.
|
||
|
||
*** ... which sets the scene ...
|
||
|
||
The bedroom group (=/groups/1= in the rule's action list) is set to
|
||
the following scene, which turns on the lights at minimum brightness:
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/scenes/7GJer2-5ahGIqz6
|
||
#+END_SRC
|
||
|
||
#+RESULTS[acf0f0fd18efd18cdf94d2c1175f148f9cd85733]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"name": "Wake Up init",
|
||
"lights": [
|
||
"2",
|
||
"3",
|
||
"5"
|
||
],
|
||
"owner": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
||
"recycle": true,
|
||
"locked": true,
|
||
"appdata": {},
|
||
"picture": "",
|
||
"lastupdated": "2018-03-11T19:46:50",
|
||
"version": 2,
|
||
"lightstates": {
|
||
"2": {
|
||
"on": true,
|
||
"bri": 1,
|
||
"ct": 447
|
||
},
|
||
"3": {
|
||
"on": true,
|
||
"bri": 1,
|
||
"ct": 447
|
||
},
|
||
"5": {
|
||
"on": true,
|
||
"bri": 1,
|
||
"ct": 447
|
||
}
|
||
}
|
||
}
|
||
#+END_SRC
|
||
|
||
*** ... and schedules the transition ...
|
||
Another schedule (=/schedules/2= in the rule's action list) is enabled
|
||
by the rule.
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/schedules/2
|
||
#+END_SRC
|
||
|
||
#+RESULTS[b408558373a001cc2354c960171b21df2669ab85]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"name": "L_04_fidlv",
|
||
"description": "L_04_fidlv_trigger end scene",
|
||
"command": {
|
||
"address": "/api/xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx/groups/0/action",
|
||
"body": {
|
||
"scene": "gXdkB1um68N1sZL"
|
||
},
|
||
"method": "PUT"
|
||
},
|
||
"localtime": "PT00:01:00",
|
||
"time": "PT00:01:00",
|
||
"created": "2018-03-11T19:46:51",
|
||
"status": "disabled",
|
||
"autodelete": false,
|
||
"starttime": "2018-03-13T10:30:00",
|
||
"recycle": true
|
||
}
|
||
#+END_SRC
|
||
|
||
/This/ schedule is a bit different from the one we saw before. It is
|
||
normally disabled, and it's time pattern (in =localtime=) is
|
||
different. The =PT= prefix specifies that this is a timer which
|
||
expires after the given amount of time has passed. In this case, it is
|
||
set to one minute (the first 60 seconds of our wake-up will be spent
|
||
in minimal lighting). Enabling this schedule starts up the timer. When
|
||
one minute is up, another scene will be set.
|
||
|
||
This one, strangely, is applied to group =0=, the meta-group including
|
||
all lights, but since the scene itself specifies to which lights it
|
||
applies, there's no real problem with it.
|
||
|
||
*** ... to a fully lit room ...
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/scenes/gXdkB1um68N1sZL
|
||
#+END_SRC
|
||
|
||
#+RESULTS[d73d2de48bbff9ac73185999ff3671ddf159dc0d]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"name": "Wake Up end",
|
||
"lights": [
|
||
"2",
|
||
"3",
|
||
"5"
|
||
],
|
||
"owner": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
||
"recycle": true,
|
||
"locked": true,
|
||
"appdata": {},
|
||
"picture": "",
|
||
"lastupdated": "2018-03-11T19:46:51",
|
||
"version": 2,
|
||
"lightstates": {
|
||
"2": {
|
||
"on": true,
|
||
"bri": 254,
|
||
"ct": 447,
|
||
"transitiontime": 17400
|
||
},
|
||
"3": {
|
||
"on": true,
|
||
"bri": 254,
|
||
"ct": 447,
|
||
"transitiontime": 17400
|
||
},
|
||
"5": {
|
||
"on": true,
|
||
"bri": 254,
|
||
"ct": 447,
|
||
"transitiontime": 17400
|
||
}
|
||
}
|
||
}
|
||
#+END_SRC
|
||
|
||
This scene transitions the lights to full brightness over the next 29
|
||
minutes (1740 seconds), per the specified =transitiontime= (which is
|
||
specified in deciseconds).
|
||
|
||
*** ... which will be switched off later.
|
||
Finally, an additional rule takes care of turning the lights off and
|
||
the wake-up sensor at 9:00 (Two and a half hours after the initial
|
||
triggering of the sensor).
|
||
|
||
#+BEGIN_SRC http
|
||
GET http://bridge/api/${username}/rules/2
|
||
#+END_SRC
|
||
|
||
#+RESULTS[68cbdf0c611d00ec6d86daa820d2dbe672f1d452]:
|
||
#+BEGIN_SRC js
|
||
{
|
||
"name": "Wake up 1.end",
|
||
"owner": "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
|
||
"created": "2018-03-11T19:46:51",
|
||
"lasttriggered": "2018-03-13T13:00:00",
|
||
"timestriggered": 2,
|
||
"status": "enabled",
|
||
"recycle": true,
|
||
"conditions": [
|
||
{
|
||
"address": "/sensors/2/state/flag",
|
||
"operator": "eq",
|
||
"value": "true"
|
||
},
|
||
{
|
||
"address": "/sensors/2/state/flag",
|
||
"operator": "ddx",
|
||
"value": "PT02:30:00"
|
||
}
|
||
],
|
||
"actions": [
|
||
{
|
||
"address": "/groups/2/action",
|
||
"method": "PUT",
|
||
"body": {
|
||
"on": false
|
||
}
|
||
},
|
||
{
|
||
"address": "/sensors/2/state",
|
||
"method": "PUT",
|
||
"body": {
|
||
"flag": false
|
||
}
|
||
}
|
||
]
|
||
}
|
||
#+END_SRC
|
||
|
||
Unlike the first rule, this one doesn't trigger immediately. It has an
|
||
additional condition on the sensor state flag using the special =ddx=
|
||
operator, which (given the timer specified) is true *two and a half
|
||
hours after* the flag has been set. As the schedule sets it at 6:30,
|
||
that means that this rule will trigger at 9:00, turn the lights off in
|
||
the bedroom, and set the sensor's flag to =false=.
|
||
|
||
** Where to go from here
|
||
|
||
The wake-up config in the phone app touched on pretty much every major
|
||
aspect of the Hue bridge API. Given the insight I now have into how it
|
||
works, I can start constructing my own schedules and transitions, and
|
||
playing with different ways of triggering them and even having them
|
||
trigger each other.
|
||
|
||
If I get around to building my rolling sunrise, I'll be sure to get a
|
||
post up on it :)
|
||
|
||
* DONE Automating My Apartment With Home Assistant :home_automation:
|
||
CLOSED: [2019-06-27 Thu 18:13]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: automating-my-apartment-with-home-assistant
|
||
:END:
|
||
|
||
A while ago, I [[relref:hue-wake-up][posted about]] my experiments with the Phillips Hue API
|
||
to create an automated morning sunrise effect. The end result was
|
||
nice, but all that mucking about with their HTTP APIs was a hassle any
|
||
time I wanted to tweak something. I wanted to define what I wanted in
|
||
a more declarative style, and have all the API calls managed behind
|
||
the scenes. [[https://www.home-assistant.io/][Home Assistant]] allowed me to do exactly that, and more.
|
||
|
||
While the Home Assistant docs are geared heavily towards setting up a
|
||
raspberry pi appliance to run everything 24/7, I don't own one, and I
|
||
already have a server going. I opted instead to get the home assistant
|
||
server running using [[https://www.home-assistant.io/docs/installation/docker/][Docker]], and setting up a git repository to hold
|
||
my configuration.
|
||
|
||
** A Brand New Day
|
||
Setting up my sunrise was actually /really/ easy. I already had the
|
||
scenes I wanted from my [[relref:hue-wake-up][previous attempt]], so it was just a matter of
|
||
codifying them in the YAML config. I split them into four scenes - a
|
||
start (dawn) and end (daylight) pair for the standing lamp at the wall
|
||
beyond the foot of the bed, and a pair for the two nightstand lights.
|
||
The end scenes include the transition time to fade in (30 minutes).
|
||
|
||
#+begin_src yaml
|
||
scene:
|
||
- name: Dawn Sun
|
||
entities:
|
||
light.standing_lamp:
|
||
state: on
|
||
brightness: 1
|
||
xy_color: [0.6042, 0.3739]
|
||
- name: Dawn Daylight
|
||
entities:
|
||
light.correls_nightstand:
|
||
state: on
|
||
brightness: 1
|
||
xy_color: [0.2376, 0.1186]
|
||
light.stephanies_nightstand:
|
||
state: on
|
||
brightness: 1
|
||
xy_color: [0.2376, 0.1186]
|
||
- name: Sunrise Sun
|
||
entities:
|
||
light.standing_lamp:
|
||
state: on
|
||
transition: 1800
|
||
brightness: 254
|
||
xy_color: [0.3769, 0.3639]
|
||
- name: Sunrise Daylight
|
||
entities:
|
||
light.correls_nightstand:
|
||
state: on
|
||
transition: 1800
|
||
brightness: 203
|
||
xy_color: [0.2698, 0.295]
|
||
light.stephanies_nightstand:
|
||
state: on
|
||
transition: 1800
|
||
brightness: 203
|
||
xy_color: [0.2698, 0.295]
|
||
#+end_src
|
||
|
||
Breaking them apart this way means I can trigger the "sun" first for a
|
||
splash of orange, then start up the nightstand "daylight" lights a
|
||
little bit later! This worked out well, too, since even at the lowest
|
||
brightness, having them turn on right at the start when the room is
|
||
totally dark had a tendency to jolt me awake. Staggering them produces
|
||
a much gentler effect. Scripting all of this took very little work...
|
||
|
||
#+begin_src yaml
|
||
script:
|
||
sunrise:
|
||
alias: Sunrise
|
||
sequence:
|
||
- service: scene.turn_on
|
||
data:
|
||
entity_id: scene.dawn_sun
|
||
- service: scene.turn_on
|
||
data:
|
||
entity_id: scene.sunrise_sun
|
||
- delay:
|
||
seconds: 180
|
||
- service: scene.turn_on
|
||
data:
|
||
entity_id: scene.dawn_daylight
|
||
- service: scene.turn_on
|
||
data:
|
||
entity_id: scene.sunrise_daylight
|
||
#+end_src
|
||
|
||
... and the end result really is quite pleasant:
|
||
|
||
#+begin_src ditaa :file ha-lights-1.png :exports none
|
||
+----+
|
||
|cA20|
|
||
| |
|
||
|{o} |
|
||
+----+
|
||
|
||
|
||
|
||
|
||
|
||
|
||
+----+ Z +----+
|
||
|cBLK| z |cBLK|
|
||
| | z | |
|
||
|{o} | |{o} |
|
||
+----+ +----+
|
||
|
||
6꞉30 AM
|
||
#+end_src
|
||
|
||
#+RESULTS[bb260c3d0d038bb1fdcd6cba2076efeff57bad80]:
|
||
[[file:static/ox-hugo/ha-lights-1.png]]
|
||
|
||
#+begin_src ditaa :file ha-lights-2.png :exports none
|
||
+----+
|
||
|cB50|
|
||
| |
|
||
|{o} |
|
||
+----+
|
||
|
||
|
||
|
||
|
||
|
||
|
||
+----+ +----+
|
||
|c327| z |c327|
|
||
| | z | |
|
||
|{o} | |{o} |
|
||
+----+ +----+
|
||
|
||
6꞉33 AM
|
||
#+end_src
|
||
|
||
#+RESULTS[07189ab694076463b9a88997b2f8be6427ea6950]:
|
||
[[file:static/ox-hugo/ha-lights-2.png]]
|
||
|
||
#+begin_src ditaa :file ha-lights-3.png :exports none
|
||
+----+
|
||
|cFFD|
|
||
| |
|
||
|{o} |
|
||
+----+
|
||
|
||
|
||
\o/
|
||
|
|
||
|
|
||
/ \
|
||
+----+ +----+
|
||
|cDDF| |cDDF|
|
||
| | | |
|
||
|{o} | |{o} |
|
||
+----+ +----+
|
||
|
||
7꞉00 AM
|
||
#+end_src
|
||
|
||
#+RESULTS[b22835dedc7d706f3f8ff6826040349decbb5d1a]:
|
||
[[file:static/ox-hugo/ha-lights-3.png]]
|
||
|
||
#+begin_center
|
||
[[file:static/ox-hugo/ha-lights-1.png]]
|
||
[[file:static/ox-hugo/ha-lights-2.png]]
|
||
[[file:static/ox-hugo/ha-lights-3.png]]
|
||
#+end_center
|
||
|
||
That just leaves the automation, which fires a half an hour before the
|
||
/actual/ sunrise, so long as the lights aren't already on and somebody
|
||
is home (using a binary sensor I defined elsewhere based on phones
|
||
detected in the house plus an override toggle).
|
||
|
||
#+begin_src yaml
|
||
automation:
|
||
- alias: Sunrise
|
||
action:
|
||
- service: script.sunrise
|
||
data: {}
|
||
trigger:
|
||
- platform: sun
|
||
event: sunrise
|
||
offset: '-00:30:00'
|
||
condition:
|
||
- condition: state
|
||
entity_id: binary_sensor.occupied
|
||
state: 'on'
|
||
- condition: state
|
||
entity_id: group.bedroom_lights
|
||
state: 'off'
|
||
#+end_src
|
||
|
||
I later extended the automation with some configuration inputs, which
|
||
tie into some new triggers and conditions. I added a "latest start
|
||
time" to make sure it always gets me up in time for me to get ready
|
||
for work, and an option to disable the wake-up on weekends.
|
||
|
||
#+begin_src yaml
|
||
input_select:
|
||
sunrise_days:
|
||
name: Days to wake up
|
||
options:
|
||
- Every Day
|
||
- Weekdays
|
||
initial: Every Day
|
||
icon: mdi:weather-sunset
|
||
input_datetime:
|
||
sunrise_time:
|
||
name: Latest start time
|
||
has_date: false
|
||
has_time: true
|
||
initial: '06:30'
|
||
automation:
|
||
- alias: Sunrise
|
||
action:
|
||
- service: script.sunrise
|
||
data: {}
|
||
trigger:
|
||
- platform: sun
|
||
event: sunrise
|
||
offset: '-00:30:00'
|
||
- platform: template
|
||
value_template: >-
|
||
{{ states('sensor.time') == (
|
||
states.input_datetime.sunrise_time.attributes.timestamp
|
||
| int | timestamp_custom('%H:%M', False)
|
||
)
|
||
}}
|
||
condition:
|
||
- condition: state
|
||
entity_id: binary_sensor.occupied
|
||
state: 'on'
|
||
- condition: state
|
||
entity_id: group.bedroom_lights
|
||
state: 'off'
|
||
- condition: or
|
||
conditions:
|
||
- condition: state
|
||
entity_id: input_select.sunrise_days
|
||
state: Every Day
|
||
- condition: and
|
||
conditions:
|
||
- condition: state
|
||
entity_id: input_select.sunrise_days
|
||
state: Weekdays
|
||
- condition: time
|
||
weekday:
|
||
- mon
|
||
- tue
|
||
- wed
|
||
- thu
|
||
- fri
|
||
#+end_src
|
||
|
||
Sprinkle in some groups, and I've got a nice panel in my Home
|
||
Assistant UI to manage everything:
|
||
|
||
#+CAPTION: The completed sunrise panel
|
||
#+ATTR_ORG: :width 800
|
||
[[file:static/images/ha-sunrise-ui.png]]
|
||
|
||
** Keep It Down!
|
||
|
||
Determined to find more things to automate, I realized that since I
|
||
have my TV audio going through a Sonos sound bar, I could very easily
|
||
automate the rather annoying ritual of leaping for the app on my phone
|
||
to turn on night mode when a movie I'm watching is getting explodey
|
||
and I realize it's a bit late in the evening to be shaking my
|
||
neighbor's walls.
|
||
|
||
#+begin_src yaml
|
||
automation:
|
||
- alias: Toggle Sonos night mode
|
||
action:
|
||
- service: media_player.sonos_set_option
|
||
entity_id: media_player.den
|
||
data_template:
|
||
night_sound: >-
|
||
{{ now().hour >= 22 }}
|
||
trigger:
|
||
- platform: time
|
||
at: '22:30:00'
|
||
- platform: time
|
||
at: '08:00:00'
|
||
#+end_src
|
||
|
||
Boom. Happier neighbors, and I can fall asleep in front of movies
|
||
without worry!
|
||
|
||
Just because I could, I also added some configurability to this
|
||
automation as well. The logic got a bit tricky, since I wanted to
|
||
configure a window that crosses a 24-hour boundary. I also added a
|
||
binary sensor so I could see when night mode was enabled from Home
|
||
Assistant.
|
||
|
||
#+begin_src yaml
|
||
automation:
|
||
- alias: Toggle Sonos night mode
|
||
action:
|
||
- service: media_player.sonos_set_option
|
||
entity_id: media_player.den
|
||
data_template:
|
||
night_sound: >-
|
||
{% set start = states.input_datetime.sonos_nightmode_start.attributes %}
|
||
{% set end = states.input_datetime.sonos_nightmode_end.attributes %}
|
||
{% set now_ = (now().hour, now().minute, now().second) %}
|
||
{% set start_ = (start.hour, start.minute, start.second) %}
|
||
{% set end_ = (end.hour, end.minute, end.second) %}
|
||
{% if start_ > end_ -%}
|
||
{{ now_ >= start_ or now_ < end_ }}
|
||
{%- else -%}
|
||
{{ now_ >= start_ and now_ < end_ }}
|
||
{%- endif -%}
|
||
trigger:
|
||
- platform: template
|
||
value_template: "{{ states('sensor.time') == (states.input_datetime.sonos_nightmode_start.attributes.timestamp | int | timestamp_custom('%H:%M', False)) }}"
|
||
- platform: template
|
||
value_template: "{{ states('sensor.time') == (states.input_datetime.sonos_nightmode_end.attributes.timestamp | int | timestamp_custom('%H:%M', False)) }}"
|
||
sensor:
|
||
- platform: time_date
|
||
display_options:
|
||
- time
|
||
input_datetime:
|
||
sonos_nightmode_start:
|
||
name: Start Night Mode
|
||
has_date: false
|
||
has_time: true
|
||
initial: '22:30'
|
||
sonos_nightmode_end:
|
||
name: End Night Mode
|
||
has_date: false
|
||
has_time: true
|
||
initial: '08:00'
|
||
binary_sensor:
|
||
- platform: template
|
||
sensors:
|
||
den_night_mode:
|
||
friendly_name: Sonos Den Night Mode
|
||
value_template: >-
|
||
{{ state_attr('media_player.den', 'night_sound') }}
|
||
#+end_src
|
||
|
||
And, voilà, a dashboard for my speakers, which I pretty much never
|
||
need to look at anymore!
|
||
|
||
[[file:static/images/ha-sonos-ui.png]]
|
||
|
||
** But Wait, There's More!
|
||
|
||
It's a too much to cover in a single blog post, but there's plenty
|
||
more going on in my config. Over time, I've tweaked and added to my
|
||
device tracking to make sure Home Assistant knows when someone's home.
|
||
I set up some text-to-speech to announce the weather in the morning,
|
||
and welcome the first person to get home. I even re-purposed an old
|
||
phone as a webcam so I can check on the cat while I'm out. My config
|
||
is on my personal gitlab server, feel free to check it out and see if
|
||
there's anything there you can use or learn from:
|
||
http://git.phoenixinquis.net/correlr/home-assistant
|
||
|
||
* DONE Trans Day of Visibility 2022 :transgender:
|
||
CLOSED: [2022-03-31 Thu]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: trans-day-of-visibility-2002
|
||
:END:
|
||
|
||
[[file:images/trans-flag-with-symbol.jpg]]
|
||
|
||
It's hard to feel positive about this year's [[https://www.glaad.org/tdov][Transgender Day of Visibility]]. On
|
||
the one hand, trans visibility is extremely important. It's because of out trans
|
||
people that I was able to understand my own identity. The more cis people really
|
||
see, talk to, and come to understand trans people, the easier it will be for
|
||
them to understand that we're, well, just /people/. Transitioning is a
|
||
/beautiful/ thing. Look at any set of photos trans people share, and you'll see
|
||
that they're not just happier, but more vibrant, more full of life, and so very
|
||
genuinely themselves! This is what folks need to see more of, and what I think
|
||
this day is meant to be about. Unfortunately, a lot of what folks are seeing
|
||
nowadays isn't trans people thriving, it's misinformation and vitriol. This
|
||
isn't at all a new phenomenon, but [[https://www.vox.com/first-person/22977970/anti-trans-legislation-texas-idaho][in recent years it's gotten overwhelming]].
|
||
|
||
This year, like last year, has brought with it a [[https://freedomforallamericans.org/legislative-tracker/anti-transgender-legislation/][record-breaking amount of
|
||
anti-trans legislation across the majority of states in the country]]. These bills
|
||
are targeting trans youths by banning them from playing sports with their peers,
|
||
forbidding any discussion about gender or queer identities in their classrooms,
|
||
requiring that trusted teachers and other school staff out them to families, and
|
||
restricting and even outlawing their healthcare. Book bans have been sweeping
|
||
the nation, intent on removing anything they consider unpleasant or
|
||
uncomfortable, which has mostly amounted to anything discussing gender,
|
||
sexuality, or race. There is a /constant/ stream of vitriol flowing across
|
||
social media and news outlets sowing outrage by [[https://www.theguardian.com/commentisfree/2018/apr/19/anti-trans-rhetoric-homophobia-trans-rights][recycling old homophobic
|
||
rhetoric]] as they label trans people as predators, anyone supporting us as
|
||
"groomers", and claim we're forcing children into life-altering surgeries. Trans
|
||
kids [[https://www.washingtonpost.com/dc-md-va/2021/04/22/transgender-child-sports-treatments/][do not get surgeries]], but laws are being pushed and passed banning them
|
||
anyway, though always with a note that [[https://www.them.us/story/trans-health-care-attacks-target-intersex-people-too][those restrictions aren't extended to
|
||
intersex kids]], who continue to be operated upon to make their bodies conform to
|
||
a comfortable binary.
|
||
|
||
Trans kids and trans adults alike, whether they're in states that are actively
|
||
arguing or passing these bills, [[https://www.thedailybeast.com/we-trans-people-will-never-surrender-but-fighting-bigots-is-exhausting?s=09&source=twitter&utm_source=pocket_mylist&via=desktop][are having to endure watching this all happen]].
|
||
Watching their identities, their /existence/ be debated, questioned, demonized,
|
||
and ridiculed. We're having to watch this all unfold, and it really feels like
|
||
[[https://truthout.org/audio/trans-youth-are-facing-right-wing-attacks-and-a-solidarity-shortage/][few people are actively defending us or standing up to this torrent of hate]].
|
||
Most of these bills aren't even getting much news coverage, and [[https://www.teenvogue.com/story/trans-people-right-wing-media?s=09&utm_source=pocket_mylist][those that are
|
||
often aren't in our favor]], framing the issues as [[https://www.bbc.com/news/uk-43255878][divisive]] or [[https://www.nbcnews.com/nbc-out/out-news/trans-swimmer-lia-thomas-speaks-scrutiny-controversy-rcna18503][controversial]]. Even
|
||
Florida's so-called [[https://www.flsenate.gov/Session/Bill/2022/1557]["Don't Say Gay" bill]] is framed first and foremost as an
|
||
attack on gay rights (which it certainly is), but leaving the very deliberate
|
||
targeting of trans kids out of the discussion. Florida governor Ron DeSantis
|
||
certainly didn't hide it, [[https://www.axios.com/dont-say-gay-bill-desantis-578593fc-5d6e-4098-b69a-c838b017ce24.html][claiming its intent is to squash so-called "woke
|
||
gender ideology"]] and
|
||
[[https://www.tampabay.com/news/florida/2022/03/29/grooming-the-buzzword-in-lgbtq-school-debate-including-in-florida/][pointing
|
||
at a large illustration from a transgender author's childrens book]] just before
|
||
signing the bill.
|
||
|
||
It's hard, as a trans person watching all of this, seeing these kids, their
|
||
parents, and ourselves under such constant and cruel attack. It's hard hearing
|
||
only the faintest murmurs of "we've got your back" from the White House as the
|
||
Equality Act continues to languish, stalled in Congress. It is hard seeing so
|
||
few people outside of the transgender community, traumatized so much by it as it
|
||
is, raising any awareness of what's going on. Each year we endure so much. We
|
||
watch public figures tell people we're perverts and predators. We watch where we
|
||
go and what we do in public, lest we inadvertently draw anyone's ire. We watch
|
||
as some trans folks do succeed, and celebrate them, but also see all the
|
||
nastiness directed at them in the media and in strangers' comments on social
|
||
media. All of this is so, so traumatizing.
|
||
|
||
Ours is a community molded by trauma and loss. Our history, vibrant as it is,
|
||
has been largely [[https://historycollection.com/16-remarkable-historical-figures-who-were-transgender/][hidden from us]] or [[https://www.teenvogue.com/story/lgbtq-institute-in-germany-was-burned-down-by-nazis][outright destroyed]]. Nearly an entire
|
||
generation of queer people was [[https://read.dukeupress.edu/tsq/article/7/4/527/168493/Trans-in-a-Time-of-HIV-AIDS?utm_source=blog&utm_medium=post&utm_campaign=j-TSQ_7-4_Feb2021][lost to hate and apathy during the AIDS epidemic]].
|
||
Many [[https://www.hrc.org/resources/fatal-violence-against-the-transgender-and-gender-non-conforming-community-in-2022][continue to be lost every year to violence]]. Mostly trans women of color,
|
||
losing their lives to hate in the rising tide of racism, misogyny, homophobia
|
||
and transphobia. We likely lose far more than we know as crimes go unreported or
|
||
misreported, as they tend to be, when trans folks [[https://chicago.suntimes.com/2021/11/29/22807775/what-i-learned-about-news-media-law-enforcement-transgender-murders-morgan-sherm-op-ed][get misgendered in death]]. This
|
||
isn't how it's supposed to be. Discovering and living as who we truly are is one
|
||
of the most joyful things in life. Being ourselves, /really sharing ourselves/
|
||
with the people we love is such a wonderful, vibrant feeling. That more and more
|
||
people are able to learn about the beautiful spectrums of identities is an
|
||
amazing thing. We've got greater resources and representation now than ever
|
||
before.
|
||
|
||
I do not believe that all of this hatred, all of these laws, any of it will win
|
||
out in the end. Trans people aren't going anywhere; being trans is just a part
|
||
of being human. We have always been here, and we will continue to be. What I
|
||
fear isn't that trans people will be wiped off the earth, because that's
|
||
impossible. What I do fear is how hard the struggle may remain for us to
|
||
continue to just /live/. I feel for these kids, terrified as they are that the
|
||
world hates them. I feel for the trans community, as we struggle with vastly
|
||
different degrees of discrimination and violence. It's a lot.
|
||
|
||
On this Transgender Day of Visibility, I feel it's important that we're not
|
||
merely seen, but seen fully. I hope that people will see our joy and our
|
||
strength and our fierce love of authentic life. I also hope that people will see
|
||
our pain, and find it in themselves to offer not just performative displays of
|
||
support but real empathy and action. We're out here showing you who we are and
|
||
what we can be. Please show /us/ who /you/ are and what we mean to you.
|
||
|
||
And, for the love of everything, [[https://www.gamespot.com/articles/jk-rowlings-anti-transgender-stance-and-hogwarts-legacy/1100-6501632/?s=09&utm_source=pocket_mylist][please leave Harry Potter in the past]].
|
||
|
||
|
||
* TODO Types in Python :programming:python:
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: types-in-python
|
||
:END:
|
||
|
||
** TODO Why Use Types?
|
||
** TODO Success Typing
|
||
** TODO Running Mypy
|
||
** TODO Specifying Types
|
||
* TODO Conference Notes: Strangeloop 2018 :programming:conferences:
|
||
CLOSED: [2018-10-04 Thu]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: strangeloop-2018
|
||
:EXPORT_OPTIONS: <:nil
|
||
:EXPORT_DATE: 2018-09-28
|
||
:END:
|
||
** Wednesday
|
||
|
||
*** Richard Feldman :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 09:00-09:30>
|
||
|
||
- Dictionaries as tables
|
||
*** Grant Maki: Facilitating Technology Change Through Cultural Change :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 09:40-10:10>
|
||
|
||
- Trying things in teams
|
||
- Tech lead impostor syndrome
|
||
- Empathizing with fear, skepticism, and choice overload
|
||
|
||
#+BEGIN_NOTES
|
||
Very interesting talk from a tech lead position. Grant emphasized
|
||
building /shared/ experiences, and approaching adoption of new tools
|
||
and ideas with empathy for the feelings of each team member.
|
||
#+END_NOTES
|
||
*** Ravi Chugh: Structured Editing for Elm in Elm :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 10:40-11:10>
|
||
|
||
- "Sketch-n-Sketch"
|
||
- Structured code editor "Deuce"
|
||
- Shows AST on code
|
||
- Apply context-aware transformations to the AST
|
||
- Type inspector mode
|
||
- Supports holes
|
||
- Live feedback for incomplete programs
|
||
- https://hazel.org - Holes for incomplete code (Cyrus Omar)
|
||
- Friday Strangeloop talk
|
||
*** Anthony Deschamps: Programming Robots with Elm :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 11:30-12:00>
|
||
|
||
- [[https://www.ev3dev.org][EV3DEV]] :: Debian distribution for LEGO Mindstorms
|
||
- Robotics is about perception, behavior, and control
|
||
- =PLatform.Worker= :: Application sans view function
|
||
*** Matthew Griffith: Building a Toolkit for Design :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 13:00-13:30>
|
||
|
||
- Legos
|
||
- Everything fits together
|
||
- Easy to build
|
||
- Easy to change
|
||
- CSS
|
||
- Dependent on lots of external factors
|
||
- Lots of gotchas
|
||
- [[https://package.elm-lang.org/packages/mdgriffith/elm-ui/latest][mdgriffith/elm-ui]] ([[https://elm-ui.netlify.com/][Website]])
|
||
- Palettes
|
||
- Consists of
|
||
- Colors
|
||
- Spacing
|
||
- Typography
|
||
- A concise set of style rules (~50 max)
|
||
- Each property defines its own tests, which can be run in arbitrary
|
||
environments
|
||
- eightshapes :: Design systems
|
||
|
||
*** Dillon Kearns: Extending Type Safety Beyond Elm's Borders :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 13:40-14:10>
|
||
- Addressing "avoidable uncertainty"
|
||
- GraphQL :: [[github:dillonkearns/elm-graphql][dillonkearns/elm-graphql]]
|
||
- Generates code from a specification
|
||
- TypeScript :: [[github:dillonkearns/elm-typescript-interop][dillonkearns/elm-typescript-interop]]
|
||
- Generates TypeScript type declarations for elm ports and flags
|
||
- Slides & Resources :: https://bit.ly/typeswithoutborders
|
||
*** Sam Row: Complex Animations Done Well :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 14:40-15:10>
|
||
|
||
- Easings
|
||
- Tracks starting and target state
|
||
- Animation updates will be jarring if the target state changes
|
||
- Springs
|
||
- Tracks current position and velocity, and target state
|
||
- Decoupling animation logic from business logic
|
||
- Separate models for application state and visual state
|
||
*** Alley Kelly McKnight: Naming Things in Elm :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 15:20-15:50>
|
||
|
||
- Categories for naming rules
|
||
- Naming conventions
|
||
- Guiding lights
|
||
- What makes a name good?
|
||
- Does the name keep us in the process flow?
|
||
- Does the name help the reader build an accurate mental model of
|
||
the code?
|
||
- Does the reader know accurately what the thing you're naming is?
|
||
- The Mental Model: Our Human Narrative
|
||
|
||
#+BEGIN_NOTES
|
||
The idea of using names to allow functions to flow as a narrative is
|
||
an interesting one, as is maintaining a consistent metaphor
|
||
throughout.
|
||
#+END_NOTES
|
||
*** Tereza Sokol: Demystifying Parsers :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Centene
|
||
:END:
|
||
|
||
<2018-09-26 Wed 16:00-16:30>
|
||
|
||
- [[https://package.elm-lang.org/packages/elm/parser/latest][elm/parser]]
|
||
- Includes support for building useful error messages
|
||
- =problem : String -> Parser a=
|
||
|
||
#+BEGIN_NOTES
|
||
I've enjoyed using Haskell's Parsec library in the past, as well as
|
||
=boost::spirit= in C++ back when I first dipped my toes into EBNF
|
||
grammars. This will be interesting to play with when I can come up
|
||
with a reason to 😊.
|
||
#+END_NOTES
|
||
** Thursday
|
||
*** Shaping our children's education in computing :session:keynote:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre
|
||
:END:
|
||
<2018-09-27 Thu 09:10-10:00>
|
||
|
||
- Project quantum (questions)
|
||
- Teals (high schools)
|
||
- Csteachers org
|
||
- Raspberry Pi foundation
|
||
- Education is rife with unintended consequences - be humble,
|
||
cooperative
|
||
*** Zero Downtime Migrations of Stateful Systems :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Pepsi Encore
|
||
:END:
|
||
<2018-09-27 Thu 10:20-11:00>
|
||
|
||
- Three different types of migrations
|
||
- perceived zero Downtime
|
||
- actually zero Downtime
|
||
- no migration of state
|
||
|
||
- Perceived Zero Downtime
|
||
- Migrating existing records
|
||
- constant syncing
|
||
- incremental
|
||
- Building confidence
|
||
- automated reconciliation and checks
|
||
- row counts
|
||
- test environment
|
||
- Decouple interactions
|
||
- allow reads
|
||
- replay writes
|
||
- Migration
|
||
- switch writes to queue
|
||
- wait for sync to complete
|
||
- switch database
|
||
- replay queue
|
||
- Actual zero Downtime
|
||
- load snapshot
|
||
- dual writes
|
||
- metrics on any issues
|
||
- No migration
|
||
- new service built aware of legacy system
|
||
- older data expires
|
||
- run in tandem
|
||
- delegate to legacy system when needed
|
||
- test with dual writes
|
||
- monitoring
|
||
- unexpected behaviors
|
||
- absent expected state
|
||
- slow roll-out
|
||
|
||
*** Chasing the Stream Processing Utopia :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Pepsi Encore
|
||
:END:
|
||
<2018-09-27 Thu 11:20-12:00>
|
||
|
||
- Architecture
|
||
- Ingestion (various sources)
|
||
- Processing (using Apache Samza)
|
||
- Complexity
|
||
- Stateless processing (filtering, /data enrichment/)
|
||
- Stateful processing (aggregation)
|
||
- Windowing
|
||
- Beam processing model
|
||
- Accuracy
|
||
- Exactly once processing
|
||
- Ingestion (de-dupe)
|
||
- Pipelined processing (Chandy Lamport)
|
||
- Storing results (idempotent writes / de-dupe)
|
||
- Scale
|
||
- Scaling ingestion
|
||
- Managed by cloud services, or yourself when self-hosting
|
||
- Scaling processing
|
||
- Challenges
|
||
- horizontal scaling
|
||
- limited to least performant component
|
||
- Typical bottlenecks
|
||
- data access
|
||
- accessing adjunct data (databases, services)
|
||
- managing temporary state
|
||
- CPU
|
||
- Any source (including batch sources, HDFS)
|
||
- Any language
|
||
- Anywhere
|
||
- Stream processing as a service
|
||
- Awesome tools
|
||
- Schema management
|
||
- Schema evolution
|
||
- ACL management
|
||
- Topic management
|
||
- Stream Processing Job Management
|
||
- Data Lineage (e.g. Apache Atlas)
|
||
- Monitoring
|
||
- Ingestion (e.g. Kafka Monitor)
|
||
- Job Progress (e.g. Burrow)
|
||
|
||
|
||
*** Rosie Pattern Language :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre - Peabody Lounge
|
||
:END:
|
||
<2018-09-27 Thu 13:30-14:10>
|
||
|
||
- Reasoning
|
||
- Regular expressions do not scale
|
||
- Combing through data to get devops insights
|
||
- libraries of named and namespaced expressions
|
||
- RPL syntax looks like a programming language
|
||
- reads/diffs like code
|
||
- executable unit tests
|
||
- macros
|
||
- RPL expressions compose
|
||
- RPL is based on parsing expression crammers, which are greedy and possessive
|
||
- makes it difficult to be accidentally inefficient
|
||
- Allows parsing recursive grammars
|
||
- Logstash (Grok) uses pattern libraries
|
||
- Can your 'grep' do this?
|
||
- =curl -s www.google.com | rosie grep -o subs net.url_common=
|
||
- Prints all matching URL sub-expressions
|
||
- =sed -n 46,49p /var/log/system.log | rosie match all.things=
|
||
- =all.things= represents a disjunction of a variety of things,
|
||
different pattern matches will be color-coded
|
||
- =head -n 1 /var/log/system.log | rosie grep -o jsonpp num.denoted_hex=
|
||
- JSON-structured output
|
||
- Formal basis
|
||
- Chomsky hierarchy
|
||
- regular - context-free - context-sensitive - recursively enumerable
|
||
- RPL (and all PEG grammars) are context-sensitive
|
||
- Shares /some/ similarities with regular expressions
|
||
- repetition (same syntax, greedy and possessive)
|
||
- character sets (simplified syntax, one name or list or range at a
|
||
time)
|
||
- Operations (look-ahead, negation, etc.) (simplified syntax)
|
||
- "choice" differs, =/= is ordered choice, possessive)
|
||
- Ships with a standard library
|
||
- Debugging
|
||
- Trace a mismatch
|
||
- =echo '17:30:4' | rosie trace time.rfc3339=
|
||
- REPL!
|
||
- Uses
|
||
- IBM uses rosie in a big-data pipeline
|
||
- Mining source code repositories
|
||
- NCSU students wrote RPL patterns to extract language features from
|
||
different languages (including bash!)
|
||
- White-listing valid input (protecting against injection attacks
|
||
per OWASP)
|
||
- Python bindings! (also, c, go), more coming
|
||
- Not very "pythonic", currently
|
||
- Resources
|
||
- http://rosie-lang.org/
|
||
- [[https://www.gitlab.com/rosie-pattern-language/rosie][rosie-pattern-language/rosie]]
|
||
|
||
*** Architecting for Active-Active Resiliency in the Cloud :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Union Station - Grand D
|
||
:END:
|
||
<2018-09-27 Thu 14:30-15:10>
|
||
|
||
- Data and Service resiliency and durability
|
||
- Basics (Ingredients)
|
||
- Availability (normal operation time / total time)
|
||
- mean time between failure (MTBF) / mean time to repair (MTTR)
|
||
- Multiple components in series (=x= and =y=) (A = Ax * Ay)
|
||
- Availability decreases
|
||
- Multiple components in parallel (multiple services =x=) (A = 1 - (1 - Ax)^2
|
||
- Availability increases
|
||
- Basic Active Architecture
|
||
- Web -> App -> DB
|
||
- cheap, simple, fast to market
|
||
- manual intervention, low resiliency, high potential rework
|
||
- Basic Active-Passive Architecture
|
||
- Load balanced app with fail-over nodes / read-replica db
|
||
- Hot or cold standby
|
||
- decreased downtime, - backup services & data, good balance
|
||
- increased costs, complexity, potential for data loss
|
||
- Basic Active-Active Architecture #1
|
||
- Multiple instances taking traffic simultaneously
|
||
- always on, minimum data loss, zero intervention
|
||
- limited tools, most complex, most expensive
|
||
- Considerations & trade-offs (Appetizers)
|
||
- Data Replication
|
||
- Active-Active (avoid this)
|
||
- multiple master databases synchronizing
|
||
- collisions
|
||
- latency
|
||
- Active-Passive
|
||
- Multiple instances writing to a single data store
|
||
- promote a read replica on failure
|
||
- Active-Active at the service level, Active-Passive at the
|
||
data level
|
||
- Easy data replication databases (managed)
|
||
- Cloud Spanner
|
||
- Cosmos
|
||
- Aurora
|
||
- Easy data replication databases (not managed)
|
||
- Apache Cassandra
|
||
- MySQL
|
||
- Considering tenants - what are we failing over?
|
||
- /Similar to considering restart strategies in OTP supervision
|
||
trees/
|
||
- As availability and durability increases, so does complexity and cost
|
||
- Resiliency patterns (The Entrees)
|
||
- Circuit breaker (stop retrying)
|
||
- Exponential back-off (slow down retries)
|
||
- Fallback (error response or cache)
|
||
- Multiple instances in different availability zones within a
|
||
region
|
||
- Worker queues and streams decouple components
|
||
- Patterns can be combined
|
||
- Tools and services (The Desserts)
|
||
- Chaos Monkey
|
||
- Serverless
|
||
|
||
#+BEGIN_NOTES
|
||
It's important to prepare for failure, and balancing availability and
|
||
consistency is always a balancing act.
|
||
|
||
For failure, I like to map out services and their dependency as I
|
||
would an Erlang supervision tree, drawing boundaries around tightly
|
||
coupled components, and defining the failure strategy for each group.
|
||
#+END_NOTES
|
||
*** Towards Language Support for Distributed Systems :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre
|
||
:END:
|
||
<2018-09-27 Thu 15:30-16:10>
|
||
|
||
#+BEGIN_QUOTE
|
||
Research is very animated! (doesn't look like it from the outside)
|
||
#+END_QUOTE
|
||
|
||
- Language support for distributed system = ways that the programming
|
||
language/compiler can aid in the development of a distributed application
|
||
- Some areas of research
|
||
- Consistency & Programming Models
|
||
- Consistency?
|
||
- Specifically programming models that provide some kind of
|
||
consistency guarantee (e.g, what guarantees do I have for a
|
||
replica of an object/datum?)
|
||
- Sometimes weak consistency is good enough
|
||
- Conflict-free replicated data types (CRDTs)
|
||
- Lasp: A Language for Distributed, Coordination-Free
|
||
Programming
|
||
- Spry is a programming model for building applications that want
|
||
to trade-off availability and consistency at varying points in
|
||
application code to support application requirements (CA in CAP)
|
||
- Mixed consistency via "Correctables" (Incremental Consistency
|
||
Guarantees for Replicated Objects)
|
||
- Give the user different APIs depending on the sort of
|
||
consistency they're after
|
||
- =invokeWeak(operation)=
|
||
- =invokeStrong(operation)=
|
||
- =invoke(operation[, levels])=
|
||
- Mixed consistency via "MixT" ([[http://www.cs.cornell.edu/andru/papers/mixt/mixt.pdf][MixT: A Language for Mixing
|
||
Consistency in Geodistributed Transactions]])
|
||
- Answers the question of what if we want to choice from
|
||
multiple consistency options in one application
|
||
- Session Types
|
||
- Can be though of as types for communication protocols
|
||
- Many variants of session types
|
||
- Binary & Multiparty (# of parties)
|
||
- Static & Dynamic
|
||
- [[http://simonjf.com/2016/05/28/session-type-implementations.html][Introduction to session types by Simon Fowler]]
|
||
- Prototypes of session types are implemented for many languages
|
||
- Erlang :: monitored-session-erlang
|
||
- Python :: [[https://www.doc.ic.ac.uk/~rn710/spy/main.pdf][SPY]]
|
||
- Haskell :: effect-sessions
|
||
- Scala :: Ichannels
|
||
- Java :: Scribble
|
||
- Static Analysis & Verification
|
||
- Others
|
||
- Whip ([[https://people.seas.harvard.edu/~chong/pubs/icfp17-whip.pdf][Higher-Order Contracts for Microservices]])
|
||
- [[http://unisonweb.org][Unison]]
|
||
- Haskell-like language for moving computations around
|
||
- Syndicate ([[https://www2.ccs.neu.edu/racket/pubs/dissertation-garnock-jones.pdf][Conversational Concurrency]])
|
||
- ML5 ([[http://www.cs.cmu.edu/~tom7/papers/modal-types-for-mobile-code.pdf][Modal Types for Mobile Code]])
|
||
- Function-Passing ([[https://infoscience.epfl.ch/record/230304/files/fp-jfp.pdf][A Programming Model and Foundation for
|
||
Lineage-Based Distributed Computation]])
|
||
- Birds-eye view of fringe projects
|
||
- [[http://christophermeiklejohn.com/publications/hotedge-2018-containers-preprint.pdf][Verifying Interfaces Between Container-Based Components (Or... A
|
||
Type System By Any Other Name)]]
|
||
- Rejected 😅
|
||
- Statically ensuring that microservices satisfy invariants -
|
||
Adelbert Chang
|
||
- Statically ensuring that functions on replicated data are
|
||
monotone - Kevin Clancy
|
||
- Fault-Tolerant Functional Reactive Programming (Functional
|
||
Pearl)
|
||
- Resources
|
||
- http://bit.ly/dist-prog-refs
|
||
*** Keynote: Erica Joy Baker - Changing the World :session:keynote:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre
|
||
:END:
|
||
<2018-09-27 Thu 16:30-17:20>
|
||
|
||
- Passionate about talking about diversity and inclusion
|
||
- It seems that everyone wants to "change the world"
|
||
- "We did it! ... Mission accomplished, in the words of George Bush.
|
||
We changed the world"
|
||
- "We got so focused on disruption that we didn't stop to think about
|
||
who or what we were disrupting"
|
||
- Recommends watching "Halt and Catch Fire"
|
||
- "Silicon Valley is Venture Capital"
|
||
- Story of "The Scorpion and the Frog"
|
||
- The VCs are the scorpions "in case that wasn't clear".
|
||
- Steps to change the world
|
||
1. Have an idea
|
||
2. Grow your ideas slowly.
|
||
3. Get customers
|
||
4. Get some more of them
|
||
5. Keep going
|
||
6. Build what you want to build on your own terms.
|
||
- Mailchimp is a sponsor (bleh), has no VC funding
|
||
- We have to ask ourselves, "I can do this, but should I do this?"
|
||
- "How we're going to change the world is by the legacy we leave
|
||
behind."
|
||
- "We must build companies that represent everyone, and all
|
||
experiences."
|
||
- Call to action: If you work in a company that you know is doing
|
||
something harmful, that you have averted your eyes from, now is the
|
||
time to speak up and do something.
|
||
- Change the world, just make sure you're changing it for the better.
|
||
** Friday
|
||
*** All the Languages Together :session:keynote:
|
||
:PROPERTIES:
|
||
:ADDRESS: Stifel Theatre
|
||
:END:
|
||
<2018-09-28 Fri 09:10-10:00>
|
||
|
||
- Different languages have different strengths
|
||
- Multi-Language software is hard
|
||
- FFI/JNI
|
||
- microservices
|
||
- Can we allow reasoning about interop at the source level?
|
||
- As opposed to running code and encountering errors / debugging
|
||
- Language specifications are incomplete - don't account for linking
|
||
- 'escape hatches' (ML/C FFI, Rust/unsafe, Java/JNI)
|
||
- We tell programmers to "be careful"
|
||
- Rethink PL Design: Principled FFIs
|
||
- Design *linking types* extensions that support safe
|
||
interoperability with other languages
|
||
- linear types (similar to rust ownership)
|
||
- pure (terminating) functions
|
||
- Only need linking type extensions to interact with
|
||
*behavior/features inexpressible* in your language
|
||
- [[https://arxiv.org/abs/1711.04559][Linking Types for Multi-Language Software: Have Your Cake and Eat It Too]]
|
||
- Reasoning about refactoring
|
||
- Let the programmer specify, do I want to only link with pure
|
||
code? Allow linking with stateful code?
|
||
- Minimal annotation burden
|
||
- Provide a default translation (e.g, a pure language defaults to
|
||
linking pure code, not stateful)
|
||
- Catching cross-language type errors
|
||
- What happens if the linked code doesn't have linking type
|
||
annotations?
|
||
- Type-preserving compilation
|
||
- Provide specifications of code invariants that can be checked at
|
||
link time
|
||
- [[https://www.cs.princeton.edu/~dpw/papers/tal-toplas.pdf][From System F to Typed Assembly Language]]
|
||
- Cross-language type errors
|
||
- Fail linking if target (compiled) types don't match
|
||
- Refactoring-Preserving Compilation
|
||
- Also known as /equivalence-preserving compilation/, /fully
|
||
abstract compilation/, and /secure compilation/
|
||
- Specifying a non-default linking type means giving up some
|
||
refactoring patterns
|
||
- Programmer gets to decide what refactorings matter to them; LT
|
||
annotations tell the compiler what linking to allow/disallow
|
||
- Linking types are the escape hatch
|
||
- Ongoing work
|
||
- Mulberry Project
|
||
- Building linking types to interop ML with Rust
|
||
- Building linking types for a pure language
|
||
- Building on top of web assembly (generating a richly typed Wasm
|
||
with state/effect encapsulation)
|
||
- What about untyped languages (Scheme, C)?
|
||
- /Untyped/ Wasm (gradual typing)
|
||
- *control* linking type
|
||
- [[http://silc.ccs.neu.edu][SILC - Secure Interoperability, Languages & Compilers]]
|
||
*** Leverage vs. Autonomy in a Large Software System :session:
|
||
:PROPERTIES:
|
||
:ADDRESS: Union Station - Grand ABC
|
||
:END:
|
||
<2018-09-28 Fri 10:20-11:00>
|
||
|
||
- Big company problems
|
||
- How do you make good use of hundreds of software engineers to
|
||
build a large system?
|
||
- How do you organize a large software system so hundreds of
|
||
engineers working on it can avoid wasting their time?
|
||
- Andy grove on org structure (High Output Management)
|
||
- functional team - increased leverage
|
||
- mission-oriented team - increased speed
|
||
- speed is the only advantage
|
||
- Conway's Law
|
||
- Software tends to look like the organization that built it
|
||
- Coplein's Law
|
||
- If your software doesn't look like your organization, you've got
|
||
trouble
|
||
- Leverage vs Autonomy
|
||
- centralized systems - increased leverage
|
||
- decentralized systems - increased autonomy
|
||
- poor leverage - waste time reinventing wheels
|
||
- poor autonomy - waste time waiting on other teams
|
||
- Costs of autonomy/decentralization
|
||
- difficult to make global improvements
|
||
- difficult to make cross-cutting changes
|
||
- uncontrolled system complexity (little consistency)
|
||
- Costs of leverage/centralization
|
||
- single points of failure / monoculture risk
|
||
- arcade knowledge (downside of knowledge specialization)
|
||
- uncontrolled system complexity (poor modularity)
|
||
- sweet spot: cluster scheduling
|
||
- increased leverage, removed repetitive operational work
|
||
- most customers want roughly the same thing
|
||
- no important loss of autonomy
|
||
- global system improvement (upgrades)
|
||
- bugs (very rarely) bring down multiple services
|
||
- sweet spot: finagle as theft service framework
|
||
- high leverage
|
||
- most customers want roughly the same thing
|
||
- no important loss of autonomy
|
||
- difficult to debug
|
||
- sour patch: monolithic service
|
||
- lots of autonomy problems (major development choke-point)
|
||
- poor sense of ownership
|
||
- very easy to cause cross-cutting bugs
|
||
- single ci/deploy pipeline
|
||
- easy to make cross-cutting changes
|
||
- good leverage
|
||
- sweet spot: service oriented architecture
|
||
- autonomy is really good
|
||
- every project needs its own ci/deployment pipeline
|
||
- difficult to cause cross-cutting bugs
|
||
- good ownership
|
||
- code is very modularized
|
||
- leverage is low
|
||
- difficult to make cross-cutting changes
|
||
- sour patch: cross-service application logic library
|
||
- where to put cross-cutting logic? attempted a shared library,
|
||
embedded in a variety of separate services
|
||
- low autonomy (lots of services to deploy with changes)
|
||
- bad ownership (which team owns this?)
|
||
- sour patch: unowned services
|
||
- consequence of autonomy
|
||
- services get lost in the shuffle between teams
|
||
- sour patch: ad-hoc service platform
|
||
- building stuff into existing services, rather than creating a new
|
||
one
|
||
- caused by an aversion to build a whole new service
|
||
- "What's the fastest way I can get this piece of work done?"
|
||
- owners of the code get a lot of leverage, but lose a lot of
|
||
autonomy
|
||
- Strato: platform for microservices
|
||
- monolithic service that hosts services inside of it
|
||
- captures some common patterns (e.g., a data store with a cache in
|
||
front of it, etc.)
|
||
- Goal is to split the systems concerns from the applications
|
||
themselves (e.g., timeouts, retries, etc.)
|
||
- Microservices within are isolated, allows modularity
|
||
- a lot easier for the code structure to match the organization
|
||
structure
|
||
- Thrift RPC
|
||
- communication between services is structurally typed
|
||
- "compatibility" relation between types makes upgrades easier
|
||
- StratoQL: a DSL for microservices
|
||
- structurally typed
|
||
- native support for thrift data
|
||
- transparent concurrency (cf. Haxl (previous Sl talk))
|
||
- large gain in leverage over separate microservices
|
||
- captures common infrastructure patterns in configuration
|
||
- autonomy for application logic owners
|
||
- many customers want roughly the same thing
|
||
- complicated interface
|
||
- centralized team a bottleneck
|
||
- difficult to debug
|
||
- downtime affects many microservices
|
||
* TODO Conference Notes: Abstractions II :programming:conferences:
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: abstractions-ii
|
||
:EXPORT_DATE: 2019-08-23
|
||
:END:
|
||
** Day One
|
||
|
||
*** Kill All Mutants
|
||
- Speaker :: Dave Aronson
|
||
- Slides :: https://bit.ly/kill-mutants-abstractions-2019
|
||
|
||
|
||
- Mutation testing
|
||
- mutates code with the goal of generating test failures
|
||
- checks for meaningful code and strict tests
|
||
- each change to our code should make at least one test fail
|
||
- at least one unit test failing on a mutation is "killing the mutant"
|
||
- First proposed in Richard Lipton's "Fault Diagnoses of computer
|
||
programs" (1971)
|
||
- Difficult to interpret results, and very CPU intensive
|
||
- Python - cosmic-ray, mutmut, xmutant
|
||
- JS - stryker
|
||
|
||
*** Maintaining a Legacy Haskell App as Not-Yet-Experts
|
||
- Speaker :: Libby Horacek
|
||
- Slides :: https://slides.com/emhoracek/haskell-24
|
||
|
||
|
||
- Migrated from a haskell app in in frames within wordpress to a
|
||
haskell app serving wordpress content via the wordpress api
|
||
- The "Genius-Oh-No Cycle"
|
||
- Chesterton's Fence
|
||
- Understand why a thing is the way it is before you try to change
|
||
it
|
||
- ~3% of functional programmers are women
|
||
- Things that helped
|
||
- Pair programming
|
||
- TDD (invert the cycle)
|
||
- RFCs
|
||
- Can be totally casual
|
||
- Sketch out a larger idea and ask for input from the team
|
||
|
||
*** Measuring the Human Impact of Software Best Practices: A Story of CSS and Empathy
|
||
- Speaker :: jnf
|
||
- Twitter :: @_jnf
|
||
- Slides :: https://speakerdeck.com/jnf/measuring-the-human-impact-of-software-best-practices-a-story-of-css-and-empathy
|
||
|
||
|
||
- "The Tyranny of Pay-as-you-go Internet"
|
||
- Worked at Mighty Ai (training data as a service), acquired by Uber
|
||
- Stuff in the talk happened before Uber
|
||
- microaggressions and microaffirmations
|
||
- "you guys" vs. (friends, folks, participants, peeps, illustrious
|
||
heroes, party people, yinz)
|
||
- microaffirmations, a guide
|
||
- https://brown.edu/sheridan/microaggressions-and-micro-affirmations-0
|
||
- active listening
|
||
- recognizing and validating experiences
|
||
- affirming emotional reactions
|
||
- micro-tasks in a webapp to generate vector images
|
||
- $0.03 to $0.05 USD
|
||
- Large base in Venezuela
|
||
- community members raised concerns about data transfer
|
||
- forgot ota internet was paid by the megabyte
|
||
- didn't know the ost per megabyte
|
||
- didn't know there was such flux in data access costs
|
||
- asked them to take a literal loss in order to participate in the
|
||
beta test
|
||
- *I assumed an experience largely consistent with my own*
|
||
- CSS is a /tax/
|
||
- 1.3MB of css vs 34KB after refactoring
|
||
- Movistar pricing: 0 MB included, each MB is $5 (5,00 Bs (bolivars))
|
||
- 92 deployments, only 9 changed the CSS, every one of them
|
||
re-generated the bundle and busted the cache
|
||
- 8 downloads @ 1.3MB (52Bs) instead of 2 at 34KB (0.34Bs) /just
|
||
for CSS/
|
||
- This is an /ethical/ issue
|
||
- ask, listen, believe, then act
|
||
*** “Testing” - You Keep Using that Word, but I Don’t Think It Means What You Think It Means!
|
||
- Speaker :: Samuel Brown
|
||
- Twitter :: @SamuelBrownIV
|
||
- Slides :: https://bit.ly/testabstract
|
||
|
||
|
||
- Testing is generally re-defined by every organization to fit their
|
||
culture and needs
|
||
- Easy to explain why an acceptance test works, harder to explain what
|
||
the utility is that you get out of a unit test
|
||
- An opinionated view of testing
|
||
- *automated* testing is the single biggest factor in determining
|
||
how fast you can ship code
|
||
- You have to build up tests in layers and spend time in the right
|
||
places
|
||
- Teams that test write code with better structure focused on
|
||
interfaces and low cyclomatic complexity
|
||
- Pit of Despair - AKA "Test Environment"
|
||
- Don't want to be dependent on the state of the environment to test
|
||
that your code works
|
||
- Useful for UI/UX exploratory testing, load testing, deployment
|
||
testing
|
||
- What can we do
|
||
- Unit tests :: Test the smallest units at the function/method level
|
||
- Integration/Component tests :: testing the composition of *two*
|
||
functional units or external dependencies that achieve a larger
|
||
operational function (*reduce* the number of variables, not
|
||
*increase*)
|
||
- E2E/Acceptance/UI/API Tests :: Testing features with all required
|
||
components integrated together /but some can be mocked/
|
||
- Test Pyramid
|
||
- low cost, fast run vs high cost, slow run
|
||
- automated unit, automated integration, automated ui/api, manual ui/api
|
||
- Fun tests
|
||
- Writing good tests can be as challenging as writing good code
|
||
- You will write more (2-3x) test code than feature code
|
||
- Testing can be cathartic
|
||
- Strategies for testing
|
||
- Legacy code by Michael Feathers
|
||
- Consider frameworks and libraries that lend themselves well to testing
|
||
- Findexamples for what you want to do
|
||
- Keep functions small and purpose-built - If it is hard to test,
|
||
it's probably too big!
|
||
- Limit conditional logic in a single function to 3 levels at most
|
||
- Test ALL error conditions respond as expected
|
||
- Use stubs, mcks and test doubles to simulate expected inputs and outputs
|
||
- Limit integration/component tests to two actors (mocks for all others)
|
||
- Create interfaces for components so that they can be mocked
|
||
- Store test-case data with your tests
|
||
- It's NEVER too late to start writing tests
|
||
- Lightning Sand - AKA Microservices
|
||
- The idea is to be (and stay) loosely coupled to the rest of the
|
||
architecture, minimizing dependencies when testing is key!
|
||
- Strategies for Microservice Testing
|
||
- Don't be QA for services you depend on - trust their interface
|
||
- Use service mocks for external dependencies
|
||
- Write testing libraries/harnesses that other teams can use
|
||
- Automate all of your tests - Without a UI you only need robots
|
||
- What about non-functional tests
|
||
- Every organization needs to evaluate whether that testing is
|
||
necessary and correct for them ($$$)
|
||
- Collect detailed metrics at service boundaries
|
||
- Abstract out service-level concerns like back-pressure, retries
|
||
and complex routing
|
||
*** How Games Can Inspire Great Application UX
|
||
- Speaker :: Scott Jenson
|
||
- Twitter :: @scottjenson
|
||
- Email :: scott@jenson.org
|
||
|
||
|
||
- *NOT* gamification (don't be on the addictive side of things)
|
||
- Video games like the word "juicy" (visual impact on input)
|
||
- Video games create tension, apps remove it
|
||
- Ralph Koster (Book: [[https://www.theoryoffun.com/][A Theory of Fun]])
|
||
- Applying the lessons in the book to UX
|
||
- Games present story, players create narrative
|
||
- Standalone features vs a journey
|
||
- The Mac story arc
|
||
- Sound
|
||
- promise
|
||
- model
|
||
- depth
|
||
- Our story isn't the user's narrative
|
||
- Games are fractal, not linear
|
||
- Games are made of games
|
||
- Each level has its own motivation, feedback, and learning
|
||
- The deeper you go, the more profound the effect
|
||
- The Learning Loop
|
||
- Intent -> Action -> Result
|
||
- Mental model -> affordance -> feedback
|
||
- "Fun is just another word for learning" - Ralph
|
||
- Mario 1-1
|
||
- move (jump over) -> opening (jump into bricks) -> attacking
|
||
(jump onto an enemie)
|
||
- "That's how we make games at Nintend: we get the fundamentals
|
||
solid first, then do as much with that core concept as our time
|
||
and ambition will allow. As forthe courses and enemies, those
|
||
actually came at the very end, they were done in a single burst
|
||
of energy..." - Shigeru Miyamoto
|
||
- Nintendo does this /all the time/
|
||
- "Desktop has much better loops than mobile" - Ralph
|
||
- Affordances
|
||
- BotW stamina wheel - training you to take the better path
|
||
- [[https://pdfs.semanticscholar.org/ce60/77a469497e464ead9f38e39ac5b0d9ec2b44.pdf][The SonicFinder, An Interface That Uses Auditory Icons]]
|
||
- Mobile import is impoverished
|
||
- Pinch
|
||
- Tap
|
||
- Long press is a hack!
|
||
- Can't find any examples of gracenotes in apps
|
||
- "It's not the flashy trick that matters, but the rigourous
|
||
application of multiple types of feedback in both important and
|
||
trivial ways" - Ralph
|
||
- Hintiness
|
||
- Hints =/= Affordances
|
||
- Affordances reinforce the loop you're on
|
||
- Hints move you to a new loop
|
||
- Navigation (Disney weenies, always seeing a central place in a
|
||
park, ensuring sight lines)
|
||
- Hintiness prevents "bottom feeding" (getting stuck at one level,
|
||
not making progress and doing new things)
|
||
- Simple, light, patient examples showing how things can be done
|
||
- Pacing
|
||
- Games work incredibly hard on the first step
|
||
- Here is a toolbox vs learn *this* first
|
||
*** Game Development in Eight Bits
|
||
- Speaker :: Kevin Zurawel
|
||
- Website :: https://famicom.party
|
||
- Slides :: https://bit.ly/gd8b-abs2
|
||
|
||
|
||
- 256x240 resolution with 64 colors
|
||
- 1 background layer, 1 sprite layer
|
||
- Backgrounds
|
||
- 30x30 grid of tiles, 1 byte per tile
|
||
- 2 pattern tables of tiles, 256 tiles each (one for sprites, one
|
||
for backgrounds)
|
||
- 64 colors. 8 of them are black (blame NTSC)
|
||
- 8 4-color palettes, 4 for sprites, 4 for backgrounds
|
||
- The first color of all pallets is the same (hardware limitation)
|
||
- Sprites
|
||
- 256 bytes of sprite ram, 4 bytes per sprite, 64 sprites at a time
|
||
- No more than 8 sprites per scanline
|
||
- solved using flickering
|
||
- Level data
|
||
- Make use of default color
|
||
- abstract elements (pipe, height /x/)
|
||
- run-length encoding
|
||
- "set decoration" - three-screens-wide default background (SMB)
|
||
- Physics
|
||
- Don't use physics (simple algorithm)
|
||
- Collision detection
|
||
- Contra uses point vs rectangle detection
|
||
- The player is always the point, where it can be hit it is a
|
||
rectangle in relation to the point
|
||
- The NES does not have a random number generator (3 optionsin
|
||
increasingorder of stupidity)
|
||
- Tetris: Do it with math (16-bit fibonacci linear feedback shift register)
|
||
- FF: (Nasir Gebelli, contractor) a lookup table of 256 random numbers
|
||
- Contra: a single global 8-bit value that increments by 7 whenever
|
||
the game is idle
|
||
- The demo uses prerecorded /actions/, it can play out differently
|
||
- Saving progress
|
||
- Password systems
|
||
- DQ2 in Japan used a "poem"
|
||
- FDS
|
||
- Shut down due to ease of piracy
|
||
- Battery-backed memory
|
||
- "Hold reset" - power issues could lead to corruption
|
||
- Write multiple times with CRC
|
||
- "Embrace the stupid"
|
||
- Is it close enough, and much more efficient?
|
||
** Day Two
|
||
*** Duolingo: Microservice Journey :ATTACH:
|
||
- Speaker :: Max Blaze
|
||
|
||
|
||
- first microservice in 2016
|
||
- making many changes to the product, many releases per day
|
||
- centralized dashboards/logging
|
||
- Terraform for infrastructure as code
|
||
- First microservice in ECS in 2017-2018
|
||
- Why move to microservices?
|
||
- Scalability problem with teams
|
||
- Slow and difficult with a monolith
|
||
- Desire to use multiple languages (monolith in python, wanting to
|
||
incorporate scala, nodejs, ...)
|
||
- Flexibility
|
||
- Velocity
|
||
- Reliability
|
||
- Cost savings
|
||
- What to carveout first?
|
||
- Not the largest chunk
|
||
- Start with a small but impactful feature
|
||
- move up in size, complexity, and risk
|
||
- consider dependencies
|
||
- First thing was the reminder service 🦉🗡
|
||
- Using circuit breakers to make microservices independent
|
||
- Why docker?
|
||
- Kind of the only game in town
|
||
- Why docker with ECS?
|
||
- task auto scaling
|
||
- task-level IAM
|
||
- needs to be supported by the aws client library (e.g., boto)
|
||
- cloudwatch metrics
|
||
- dynamic alb targets
|
||
- manageability
|
||
- Microservice abstractions at Duolingo
|
||
- Abstracted into terraform modules
|
||
- Web service (internal or external)
|
||
- load balancer and route 53
|
||
- worker service (daemon or cron)
|
||
- sqs and event-based scheduling
|
||
- data store
|
||
- monitoring
|
||
- CI/CD
|
||
- Github -> Jenkins -> ECR/Terraform (S3) -> ECS
|
||
- Load balancing
|
||
- ALB vs. CLBs
|
||
- ALBs more strict when handling malformed requests (defaults to
|
||
HTTP/2 (headers always passed in lowercase)
|
||
- Differences in cloudwatch metrics (continuous in CLBs, discrete
|
||
in ALBs)
|
||
- Standardizing microservices
|
||
- develop a common naming scheme for repos and services
|
||
- autogenerate as much of the initial service as possible (?)
|
||
- move core functionality to shared base libraries
|
||
- *provide standard alarms and dashboards*
|
||
- /periodically review microservices for consistency and quality/
|
||
- Monitoring microservices
|
||
- includes load balancer errors
|
||
- pagerduty integration
|
||
- includes links to playbooks
|
||
- emergency pages, warnings go to email
|
||
- schedules and rotations are managed by terraform
|
||
- Grading microservices
|
||
- Cost reduction options
|
||
- Cluster
|
||
- instance type
|
||
- pricing options
|
||
- auto scale
|
||
- add/remove AZs
|
||
- using "Spot" (spotinst) to save money on ephermeral cluster instances
|
||
- drains ECS services
|
||
- spreads capacity across AZs
|
||
- bills on % of savings
|
||
- ECS allows oversubscription of memory, *WE DO NOT RECOMMEND THIS*
|
||
- AWS Limits
|
||
- EC2 has a hard-coded maximum # of packets(1024/s) sent to an amazon
|
||
provided dns server
|
||
- Nitro is not caching DNS requests where Xen was
|
||
*** Mentoring the way to a diverse and inclusive workplace
|
||
- Speaker :: Alexandra Millatmal
|
||
- Twitter :: @halfghaninNE
|
||
- Email :: hello@alexandramillatmal.com
|
||
- Slides :: http://alexandramillatmal.com/talks
|
||
|
||
|
||
- Developer at Newsela (Education tech company promoting literacy in
|
||
the 2nd-12th grade space)
|
||
- The tenants of mentorship are similar to the tenants of inclusive
|
||
companies
|
||
- Mentorship doesn't work for folks of under-represented backgrounds
|
||
- Finding very similar entry level jobs, very homogenous teams, no time to
|
||
support learning
|
||
- Skill-building and diversity appear related
|
||
- What if strong mentorship /begets/ diversity & inclusion?
|
||
- Good mentorship / good diversity
|
||
- Supporting and retaining engineers with underrepresented identities
|
||
- be welcoming and inclusive in the recruiting process
|
||
- post openings on "key values"
|
||
- company must list their values
|
||
- conveys people and tech values on the same level
|
||
- candidates filter jobs based on their prioritized values
|
||
- referrals can cut both ways
|
||
- can increase the homogenous nature of the workplace
|
||
- maybe direct referall bonuses into donations to inclusive
|
||
groups that attract and talent
|
||
- affinity groups
|
||
- caucuses across departments working with management
|
||
- standardized review process
|
||
- stanford research into review proceseses
|
||
- men overrepresented in the higher tier, women in the middle
|
||
- standardizing removed the bias
|
||
- clear definitions of roles and responsibilities
|
||
- do they have ownership
|
||
- are these employees getting a seat at the table for decisions
|
||
- representation in leadership
|
||
- are there people there that look like me?
|
||
- is there a clear model of advancement? allyship in leadership?
|
||
- investment in internal & external advocacy
|
||
- signals that copmanies understand the systematic barriers to
|
||
inclusion and diversity
|
||
- sponsorship - "super mentorship"
|
||
- stark differences in valuation of the above bulletpoints between
|
||
underrepresented groups and well-represented groups (women vs men,
|
||
lgbt+ vs straight men)
|
||
- Supporting and leveling up junior engineers
|
||
- recruiting process / relationships
|
||
- the candidate should be receiving feedback on their performance in
|
||
the recruting process!
|
||
- Gives them constructive advice and context
|
||
- apprenticeships and clearly defined entry-level positions
|
||
- is there a clear path for growth?
|
||
- clear and structured onboarding
|
||
- please do not make their point person a person they report to
|
||
- need to go to information from somone that doesn't involve
|
||
company politics
|
||
- information should exist outside of leads/managers heads
|
||
- define onboarding procedures in a shared space
|
||
- learning groups
|
||
- space to ask questions and demonstrate leadership, particularly
|
||
with peer-to-peer learning
|
||
- formalized mentorship
|
||
- ensure that compensated time is resulting in measurable goals
|
||
for the junior engineer
|
||
- recommend them for opportunities
|
||
- standardized review process
|
||
- reframe junior-ness as an opportunity, not a deficit of skill
|
||
- Mentorship with diversity and inclusion in mind
|
||
- this work is really hard
|
||
- easy to fall into a pattern of saying you're making progress
|
||
without measuring to make sure that's the case
|
||
- intent is only half of the picture
|
||
- the other half is /sacrifice/ to make real, measured investments
|
||
- mentorship should begin during the interviews
|
||
- [[https://www.wired.com/story/for-young-female-coders-internship-interviews-can-be-toxic/][wired article on young women's interview experiences]] (today?!)
|
||
- place serious focus on developing mentors
|
||
- forces mentees to manage /up/
|
||
- mentorship is a two-way street
|
||
- have you ever seen someone become a better collaborator after
|
||
mentoring a junior engineer?
|
||
- mentorship is leadership and it's learned
|
||
- have clear growth objectives for the mentor and the mentee
|
||
- mentorship should happen on compensated time
|
||
- rethink the peer group
|
||
- slack channel for juniors spread across different offices
|
||
- wasn't an organic space to share knowledge
|
||
- a black junior woman engineer's peers aren't just other black
|
||
employees, or women, or other limited groups
|
||
- What's the value to the company?
|
||
- make a business case for mentorship
|
||
- that will drive diversity and inclusion
|
||
- mentorship can
|
||
- build brand halo among candidates
|
||
- distribute management responsibilities
|
||
- build its own workforce
|
||
- distributes business knowledge working on real business projects
|
||
- fosters relationship building and belonging
|
||
- practices wielding expertise, fosters bonding over work
|
||
|
||
*** Sextech: The good, the Bad & the Bias
|
||
- Speaker :: Alison Falk
|
||
- Twitter :: alisonfalkpgh
|
||
- Slides :: https://bit.ly/MillatmalAbstractionsII
|
||
|
||
|
||
- Deepfakes
|
||
- people onforums requesting deepfakes of coworkers, etc.
|
||
- takes few photos
|
||
- women are most typically targeted
|
||
- silenced, made less credible due to the video
|
||
- no criminal recourse
|
||
- currently thriving
|
||
- nonconsensual / revenge porn
|
||
- laws are vague (interstate implications)
|
||
- sexting & development of sexual identity not reflected in the law
|
||
- this gets caught under the net of sharing child pornography
|
||
- Just because you can doesn't mean you should
|
||
- harms sex workers
|
||
- sex trafficing is only 19% of human trafficing
|
||
- 25% of sex workers sexually assaulted by officers
|
||
- multiple times during stings
|
||
- arrests inflate statistics
|
||
- 90.8% of victims are deported
|
||
- Bias
|
||
- don't need to reach orgasm to procreate? considered a vice, not
|
||
family planning
|
||
- approved / not approved.com
|
||
- approved vs non approved ads
|
||
- silencing of minority / repressed groups
|
||
- payment processors
|
||
- sex industry is the first adopter of new tech
|
||
- kicked off, considered a liability
|
||
- shadowbanning
|
||
- facebook's recent patented content filtering algorithm
|
||
- SESTA/FOSTA
|
||
- doesn't punish traffickers, makes websites legally liable for any
|
||
user generated content found to "knowingly assist, facilitate, or
|
||
support sex trafficing"
|
||
- interferes with sex education
|
||
- pushes sex trafficking further underground
|
||
- based on moral panic (National Center on Sexual Exploitation,
|
||
formerly Morality in the Media)
|
||
- hurts the most vulnerable in our community
|
||
- Root of the problem
|
||
- education <=> laws -> media/tech
|
||
- who receives accurate sex education
|
||
- searches on sex tech aren't showing any of these issues
|
||
- teach children to know about their bodies so they don't allow
|
||
others to make decisions about their bodies
|
||
- sextechspace
|
||
- resources online
|
||
- What can you do today
|
||
- support orgs like @decrimNY, @decrimNowDC, @TheBADASS_army
|
||
- use your platform to spread awareness
|
||
- advocate for agency and consent
|
||
- make sure all stakeholders are at the table
|
||
- "If you are not intentional about being inclusive, what you will do
|
||
is perpetuate exclusion"
|
||
|
||
*** Passing the Torch Without Dropping The Ball: Lessons in Documentation
|
||
- Speaker :: Mary Thengvall, Rain Leander
|
||
- Twitter :: @mary_grace, @rainleander
|
||
|
||
|
||
- TL;DR: It's not just a question of clocking out and handing over the
|
||
keys
|
||
- Advocato shirts!
|
||
- Getting sick, switching roles, moving on
|
||
- Why should you care about a transition plan?
|
||
- Taking time off without worrying
|
||
- Transitioning into a promotion
|
||
- Delegate: documented things allow other people to volunteer
|
||
- Are you going to do that forever? document it and let someone else
|
||
take over
|
||
- The handover document
|
||
- the overview
|
||
- project goals
|
||
- reading list
|
||
- dramatis personae
|
||
- who are the stakeholders/elders?
|
||
- learn from them, and document!
|
||
- who's the quiet person that gets stuff done?
|
||
- who needs a bit more time to ramp up?
|
||
- knowledge needs to be documented and disseminated with care,
|
||
perhaps a bit private
|
||
- not toxic, just necessary
|
||
- the regular tasks
|
||
- wish list
|
||
- that which remains undone
|
||
- the inventory
|
||
- budget
|
||
- credentials
|
||
- the keys to the castle
|
||
- +eDit edIt+ Edit
|
||
- what's no longer accurate?
|
||
- what's missing?
|
||
- what's confusing?
|
||
- revisit it, keep it fresh
|
||
- Prioritize
|
||
- what's urgent?
|
||
- what's important?
|
||
- what can only you do?
|
||
- what's the low-hanging fruit that's easy to ramp up on?
|
||
- What works for you?
|
||
- don't be scared to make changes
|
||
- or to say no
|
||
- be sure the community agrees with your changes
|
||
- Share your stories
|
||
|
||
*** The Times They Are a-Changin': A Data-Driven Portrait of New Trends in How We Build Software, Open Source, & What Even is "Entry-Level" Now
|
||
- Speaker :: Heather Miller
|
||
- Twitter :: @heathercmiller
|
||
|
||
|
||
- Tracking the shift in focus in open source Scala
|
||
- Things that are changing fast
|
||
- how we build software
|
||
- open source
|
||
- our idea of software engineers
|
||
- what should they know?
|
||
- How people are getting into tech
|
||
- hiring is difficult
|
||
- there's a massive gap between jobs available and people to fill
|
||
them
|
||
- a large portion of professional developers are new
|
||
- we need to adapt, culturally, to make room for lots more newcomers
|
||
- frameworks and reuse to reduce friction?
|
||
- existing devs are burning out
|
||
- "With companies unable to fill open positions, current employees
|
||
are expected to fill in the gaps"
|
||
- Increased diversity would help
|
||
- Also immigration, remote workers
|
||
- [[https://dl.acm.org/citation.cfm?id=2702549][Paper: Gender and tenure diversity in github teams]]
|
||
- increased diversity = increased productivity
|
||
- How do we stop people from disengaging?
|
||
- Women disengage earlier than men
|
||
- Open source adoption
|
||
- Dramatically increased since 2010
|
||
- Open source became the default choice
|
||
- Low-cost with no vendor lock-in
|
||
- Open source components exist in 96% of applications scanned and
|
||
analyzed by Synopsys in 2018, with an average of 257 components
|
||
per application. 36% of code bases were open source components in
|
||
2017, 57% in 2018.
|
||
- OSS projects 62% self-funded, 49% employer-funded
|
||
- "Truck factor"
|
||
- 64% of top projects on Github relied on 1-2 devs to survive
|
||
- Ecosystem and community are everything
|
||
- Leo Meyrovich - [[https://dl.acm.org/citation.cfm?id=2509515][Paper on programming language adoption]]
|
||
- Most important factor is an ecosystem / open-source libraries
|
||
- Professional developers want an active community
|
||
- All these puzzle pieces need polishing
|
||
|
||
** Day Three
|
||
|
||
*** Analysis of the Feeling of the Attendees to a Talk in Real Time
|
||
- Speaker :: Juantomás Garcia Molina
|
||
- Twitter :: juantomas
|
||
|
||
|
||
- Google cloud functions triggered by Google Storage, Pub/Sub, or REST
|
||
- Using google vision API to detect facial featuers and emotional
|
||
responses
|
||
|
||
*** Identity Expression: Coming Out of the Work Closet
|
||
- Speaker :: Matthew Rogers
|
||
|
||
|
||
Coming out of the Closet
|
||
- Coming out doesn't just happen once, it happens a /lot/
|
||
- "Spheres of existence" you have to come out to
|
||
- Friends
|
||
- Peers
|
||
- Family
|
||
- Digital
|
||
- Public
|
||
- *Work*
|
||
- Story
|
||
- The comment is made
|
||
- Affected how I work, which affects my coworkers
|
||
- Negative effect spreads
|
||
- Expands further while training/interacting
|
||
- It isn't always cut and dry
|
||
- The fear of "what if" can be damaging all its own
|
||
- In 30 states, LGBTQ+ workers aren't fully protected under the law
|
||
- PA included
|
||
- Professinal risks
|
||
- Fired outright
|
||
- Passed over for promotions or projects
|
||
- Refusal of reference for next job
|
||
- Personal risks
|
||
- Uncomfortable or dangerous work environment
|
||
- lost source of income
|
||
- Forced to change field or location
|
||
- Ther's risk to the business witself when employees hide part of
|
||
their identity
|
||
- Productivity
|
||
- Say that 10% of your day goes towards identity concealment
|
||
- adds up to 6 weeks of lost productivity every year
|
||
- Communication
|
||
- less likely to engage
|
||
- avoid people
|
||
- crosses levels
|
||
- Creativity
|
||
- stress + anxiety
|
||
- requires vulnerability
|
||
- Collaboration
|
||
- needs creativity + communication
|
||
- not getting the best work
|
||
- What if things went the other way?
|
||
- Productivity :: fewer distractions from your work
|
||
- communication :: problems caught, efficiency goes up
|
||
- creativity :: flourishes in safe + comfortable environments
|
||
- collaboration :: becomes easier and more routine
|
||
- Why focus on such a small group of people
|
||
- ~4.5% of the entire population
|
||
- Just a method of expressing identity, not limited to just queer
|
||
people
|
||
- 4.5% queer
|
||
- 6% practicing non-christian faiths
|
||
- 19% have a disability
|
||
- 22% persons of color
|
||
- 47% female
|
||
- Isn't this just about feelings?
|
||
- this holds personal and professional importance with real mental
|
||
healthy implications
|
||
- Your "Selves"
|
||
- Private
|
||
- Home
|
||
- *You*
|
||
- Your core self
|
||
- Public
|
||
- Work
|
||
- Dissonance
|
||
- Your genuine self suffers as you put your energy into maintaining
|
||
a separate self or concealing part of your identity
|
||
- Personal Consequences
|
||
- drives down those four factors
|
||
- is mentally exhausting
|
||
- it becomes a cycle
|
||
- Business consequences
|
||
- Quality of Product
|
||
- Employee Satisfaction
|
||
- Company Culture
|
||
- Bottom Line
|
||
- What can I do now?
|
||
- Introduce yourself using your pronouns
|
||
- Says you care how other people want to be addressed
|
||
- Start ERGs (Employee resource groups)
|
||
- People with shared experiences
|
||
- Acknowledge and celebrate Black History Month, Women's Month,
|
||
Pride Month, etc. Let people feel *seen*
|
||
- Don't ask to touch anyone's hair. *Ever*.
|
||
- Accommodate variations in holidays and scheduling around religious
|
||
practices
|
||
- *Don't get embarrassed* if you mess up. Apologize, correct
|
||
yourself, and learn.
|
||
- Replace words like "wife/husband/boyfriend/girlfriend/etc" with
|
||
partner
|
||
- Be careful asking about personal and private relationships in
|
||
general
|
||
- Look into unconscious bias training. Use HR.
|
||
- Look around the room
|
||
- Different perspectives and experiences simply work better
|
||
- When you don't know, research. Ask questions, if you must. Just
|
||
*don't assume*.
|
||
- If you are in the closet at work, *open the door a little*
|
||
- Make sure you're not blocking someone else's exit
|
||
|
||
*** Overcoming Challenges: An Attitude of Iteration
|
||
- Speaker :: Eric Johnson
|
||
|
||
|
||
- "The Look" (first date story)
|
||
- If you're different and it shows, it's assumed every problem in the
|
||
book belongs to you
|
||
- "You're not good enough for her" or "That's the saddest thing I've
|
||
ever seen"
|
||
- (context: one finger on each hand, one toe on each foot, 4/5 kids
|
||
share this trait)
|
||
- "Man, I'm /different/"
|
||
- "Things are gonna be a little /challenging/ for me, aren't they?"
|
||
- *We are all challenged*
|
||
- How you deal with that challenge is the model by which everyone else
|
||
is going to deal with it
|
||
- 50/50 chance of the genetic condition being passed on (5 fingers, or
|
||
/something else/)
|
||
- /You/ have to make the choice of how to deal with it
|
||
- Can't force children to make a decision, they have to make it on
|
||
their own
|
||
- /It is a blessing, not a curse/
|
||
- /It is what it is/
|
||
- It'll never be what it isn't
|
||
- No excuses
|
||
- "I can't do x /yet/", not "I can't do x"
|
||
- We have to adapt
|
||
- Never tell kids they "can't do x",but "it's gonna be hard, you're
|
||
gonna have to figure it out"
|
||
- Have fun
|
||
- Sometimes life is hard, but you've still gotta go on. you try again,
|
||
and you try again.
|
||
|
||
*** Beyond The Sunset: How To Wring The Maximum Joy From Your Last 10 Years In Tech :ATTACH:
|
||
- Speaker :: Vanessa Kristoff
|
||
- Twitter :: @vanessakristoff
|
||
- Email :: Vanessa.kristoff@gmail.com
|
||
|
||
|
||
- "Pre-tirement"
|
||
- 55 years old
|
||
- Ageism is real
|
||
- "Will they fit the culture" BS
|
||
- But Don't be "that person"
|
||
- "better back in the day"
|
||
- Job satisfaction over time
|
||
- U-shaped curve
|
||
- http://www.andrewoswald.com/docs/jooparticle.pdf
|
||
- Imposter Syndrome
|
||
- No, seriously, for reals, after all these years, you are NOT an
|
||
imposter
|
||
- Hard to have fun while struggling with this
|
||
- We are the Village Elders
|
||
- That comes with responsibility
|
||
- We should be helping people
|
||
- It's just a job
|
||
- You should know that by now
|
||
- Sponsor a Newbie
|
||
- Teach your coworkers what grace is, by example
|
||
- Sponsor vs mentoring
|
||
- mentoring is "tweaking" them to fit in
|
||
- sponsoring is promoting them
|
||
- Give whimsical talks at conferences
|
||
- Optional: create tag clouds referencing CORBA
|
||
- You have knowledge that not everybody has
|
||
- Pretend to be an Extrovert
|
||
- Doing so will help the folks you're extroverted with
|
||
- This is why that imposter thing is important
|
||
- Sign up for the messiest possible coding
|
||
- what do you have to fear? NOTHING
|
||
- Don't stop learning
|
||
- Volunteer to write UI code or backend code or try that Haskell
|
||
thing
|
||
- Join a new industry
|
||
- Use your network for good
|
||
- You have contacts
|
||
- Your people need contacts
|
||
- See how that works?
|
||
- Gossip and complaining
|
||
- It's not adding value to your work life, so just stop it
|
||
- Make a plan
|
||
- start NOW thinking about what you'll do
|
||
- how will you fill/structure your time?
|
||
- you will need a social life once you're retired
|
||
- Get your house in order, literally and figuratively
|
||
- Marie Kondo that crap
|
||
- Use your employer health care (if you have it) while you can
|
||
|
||
*** How Live Coding Changed My Life
|
||
- Speaker :: Jesse Weigel
|
||
- Twitter :: @JesseRWeigel
|
||
|
||
|
||
- Volunteer live coder for freeCodeCamp
|
||
- How I got started
|
||
- Paid for private github repos rather than letting folks see my code
|
||
- "I make a lot of mistakes and I am a full time dev. Maybe it would
|
||
be helpful for beginners to see that."
|
||
- First streams were really bad and nobody watched
|
||
- Asked for help on the freeCodeCamp forum
|
||
- The community
|
||
- all ages, locations, and skill levels
|
||
- eager to learn and contribute
|
||
- overwhelmingly positive
|
||
- always learning and trying new ways to do things
|
||
- VERY diverse watchers
|
||
- Consistency is key
|
||
- Set time for the streams
|
||
- Keeping it positive
|
||
- Thank for negative feedback
|
||
- Ask for clarification if it's not constructive ("what can I do
|
||
better?")
|
||
- People with situational anxiety, etc. feeling safe
|
||
- Video about depression, mental health issues
|
||
- What I've learned
|
||
- So much collaboration
|
||
- Pull requests!
|
||
- Confidence building
|
||
- Viewers too, gaining confidence over time
|
||
- Viewers get jobs!
|
||
- How you can start
|
||
- Make a youtube or twitch channel
|
||
- Broadcast and share your screen (OBS)
|
||
- Start coding!
|
||
- Advice
|
||
- Adapt based on feedback
|
||
- show your mistakes
|
||
- show your face
|
||
- don't be afraid to say I don't know
|
||
- It's okay to say nothing (or better yet thank you) when someone
|
||
tells you something that you already knew
|
||
- it may not be new to them, they could be excited and want to
|
||
help
|
||
- Always be positive and encouraging
|
||
- Give encouragement, not solutions
|
||
- Things to avoid
|
||
- talking badly about another language, framework, library,etc
|
||
- laughing at a question (assume every question is serious)
|
||
- getting angry
|
||
- coding and reading the chat at the same time
|
||
- negativity (there is already enough negativity in the world)
|
||
|
||
*** Debugging Our Feelings :ATTACH:
|
||
- Speaker :: Jamie Strachan
|
||
- Twitter :: @JamieStrachan
|
||
|
||
|
||
- Working at odds with professional development and depression
|
||
- Cognitive Behavioural Therapy
|
||
- "Feeling Good" by Dr. David Burns
|
||
- parallells with developer life
|
||
- Experiences -> 👩 Thoughts -> Responses
|
||
- "There is nothing either good or bad, but thinking makes it so" (Shakesspeare)
|
||
- We can change how we feel if we change how we think
|
||
- Input -> 💻 Code -> Output
|
||
- Starting a project, getting overwhelmed, distracted, guilty, nervous
|
||
- Similar to application in an incorrect, incomplete state. Not what
|
||
we wanted yet.
|
||
- We test our output
|
||
- We trust our feelings
|
||
- Emotional reasoning
|
||
- Because I feel something, it must be true
|
||
- Not rational
|
||
- Would be like looking at the app in that state, "I guess I'm
|
||
done."
|
||
- Feelings should be
|
||
- +positive+
|
||
- helpful
|
||
- reasonable
|
||
- rational
|
||
- good at /rationalizing/, working our way /back/ from the
|
||
feelings, not super helpful
|
||
- Mind reading
|
||
- We respond to what /we/ think /they're/ thinking
|
||
- Leads to impostor syndrome
|
||
- Test the feelings instead. Ask the question, don't guess.
|
||
- All-or-nothing thinking
|
||
- Perfectionism
|
||
- Test
|
||
- Antiperfectionism sheet
|
||
- Task, Effectiveness, Satisfaction
|
||
- Testing helps identify problems, but doesn't fix them
|
||
- Code won't get any better without changes
|
||
- Editor for thoughts
|
||
- Notebook
|
||
- two column technique
|
||
- left hand side: automatic thoughts
|
||
- thought patterns that are causing us to feel that way
|
||
- "I'm overwhelmed" -> thinking, "this is going to be hard"
|
||
- "Guilty" -> thinking, I'm a procrastinator
|
||
- Nervous -> thinking, I'm not going to finish this on time
|
||
- right hand side: rational responses
|
||
- fortune telling
|
||
- we've predicted the future, and are responding as though
|
||
it's true
|
||
- I can't know this
|
||
- it could also be fun, educational
|
||
- labeling
|
||
- fancy term for name-calling
|
||
- reduces people to one trait
|
||
- "I've always been a procrastinator, I'll always be a procrastinator"
|
||
- I'm not any one thing (I'm procrastinatING)
|
||
- I don't always procrastinate
|
||
- fortune telling
|
||
- takes away our autonomy (it's inevitable)
|
||
- I have control over the outcome
|
||
- there are other options
|
||
- Thoughts can have a "home field advantage" in our heads
|
||
- How does this change when it comes from someone else?
|
||
- It doesn't
|
||
- Still just an experience, just your thoughts
|
||
- "No one can make you feel inferior *without your consent*" -
|
||
Eleanor Roosevelt
|
||
- Magnification and Minification
|
||
- Disqualifying the postitive
|
||
- Accepting a compliment
|
||
- pause
|
||
- say thank you
|
||
- Using the two-column technique, have someone to role play them with
|
||
- We don't always extend sympathy to ourselves
|
||
- "Should" is the worst word in the english language
|
||
- Try "I /want/..."
|
||
- Test your feelings
|
||
- Get your thoughts out of your head
|
||
- Get help
|
||
** Not Attended
|
||
*** The Mental Impact of Tech Interviews
|
||
- Slides :: https://docs.google.com/presentation/d/1y4SwAuJcYgCJ-5lSpF560QBWaMBIrlLFA5BrMH-UKFw/edit?usp=drivesdk
|
||
- Speaker :: Zack Zlotnik
|
||
* TODO Doing the next right thing
|
||
I'm really good at beating myself up.
|
||
|
||
For me, doing the next right thing means:
|
||
- Taking a little extra time in the mornings to heat up some water for tea
|
||
- Preparing myself a bubble bath and relaxing quietly for an hour listening to a
|
||
favorite podcast
|
||
- Getting out for a short walk when the weather's good after a few days of
|
||
feeling bad for not exercising
|
||
- Writing in my journal when I'm feeling crappy about something to get my
|
||
feelings out
|
||
- Watching a favorite episode of a show when I need to feel comforted
|
||
* TODO When to sanitize and when to validate :security:programming:
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: when-to-sanitize-when-to-validate
|
||
:EXPORT_DATE: 2021-10-14
|
||
:END:
|
||
|
||
Data should be validated coming into your application, and sanitized when going
|
||
out.
|
||
|
||
For the purpose of this article, I'll work with the following definitions:
|
||
- Validation :: Checking data against the expected types and constraints of a
|
||
domain, rejecting data that does not comply.
|
||
- Sanitization :: Transforming data to satisfy constraints (e.g. removing
|
||
unwanted characters or escaping them) in order to allow the data through.
|
||
|
||
The part of your system that is accepting data should not need to be concerned
|
||
with how it /may/ be output elsewhere. If you can ensure that your output is
|
||
always properly sanitized, you can avoid needlessly complicating your validation
|
||
logic, thereby keeping your input and output layers neatly decoupled.
|
||
|
||
** Reasons to sanitize output
|
||
*** Cross-site scripting (XSS) prevention
|
||
*** SQL injection prevention
|
||
* TODO I didn't want to glue my new phone mount to my dashboard :3d:printing:
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: dashboard-phone-mount
|
||
:EXPORT_DATE: 2022-06-20
|
||
:END:
|
||
|
||
What do you do when you want to mount your phone in your car but it has trouble
|
||
staying put on your dashboard? Well, if you're me, you try your hand at
|
||
designing a counterweight and fabricate it with a 3D printer you're borrowing
|
||
from a friend.
|
||
* DONE Automating our garage door with an ESP2866 and Home Assistant :home_automation:electronics:
|
||
CLOSED: [2022-10-24 Mon 15:57]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: automating-our-garage-door
|
||
:EXPORT_HUGO_CUSTOM_FRONT_MATTER: :images '("/ox-hugo/garage-door-feature.png")
|
||
:END:
|
||
|
||
[[file:images/garage-door-feature.png]]
|
||
|
||
Now that I've got a house again, I can really start playing with home automation
|
||
projects a lot more. The first thing I plan to do is come up with something to
|
||
monitor and automate the garage door, as I've /already/ panicked from not
|
||
remembering whether I'd closed it when leaving the house. It turned out I had
|
||
closed it, but I can definitely do without that anxiety. What better way to
|
||
remove that anxiety than by being able to remotely monitor and control the door!
|
||
|
||
** Choosing the hardware
|
||
Controlling a garage door is a bit more involved than automating smart lights,
|
||
so it was time to finally dip my toes into wiring up and programming some
|
||
microcontrollers. I did some research and settled on the popular [[https://en.wikipedia.org/wiki/ESP8266][ESP8266]] series
|
||
of microcontrollers, and found myself a set of WeMos D1 mini clones with
|
||
built-in micro USB connectors ([[https://www.amazon.com/gp/product/B081PX9YFV][$3 USD each on Amazon]]). I also snagged myself a
|
||
heavy-duty looking [[https://en.wikipedia.org/wiki/Reed_switch][reed switch]] to monitor when the door is closed ([[https://www.amazon.com/Magnetic-Contacts-Shutter-Adjustable-Bracket/dp/B07ZBT28L8][$17 USD on
|
||
Amazon]]), and a pack of 3 volt DC single-channel optocoupler relays ([[https://www.amazon.com/Cermant-Channel-Driver-Module-Optocoupler/dp/B0B4MS62X6][$5 USD
|
||
each on Amazon]]). I chose single-channel as I have only one door, getting
|
||
modules with more than one channel could make it easier to hook everything up if
|
||
you have more. Because this is my first electronics project, I also grabbed
|
||
myself an [[https://www.amazon.com/gp/product/B073ZC68QG][electronics kit]] with a breadboard, jumper wires, and a bunch of fun
|
||
components to fiddle around with. I tacked on some [[https://www.amazon.com/gp/product/B071H25C43][USB cables]] and [[https://www.amazon.com/gp/product/B0794WT57Y][power bricks]]
|
||
for powering my creations as well.
|
||
** Choosing the software
|
||
There are multiple options for developing the firmware to install on the ESP8266
|
||
controller. After looking at the [[https://www.arduino.cc/en/software/][Arduino IDE]] and [[https://www.nodemcu.com/index_en.html][NodeMcu]] as possible development
|
||
options, I settled on using [[https://esphome.io/][ESPHome]] as it is super simple to set up (Arduino
|
||
coding looks fun, but I'll get everything I need just using some [[https://en.wikipedia.org/wiki/YAML][YAML]]
|
||
configuration) and it integrates super easily with [[https://www.home-assistant.io/][Home Assistant]] (the platform
|
||
I use for all of my home automation). I was able to get up and running just by
|
||
[[https://esphome.io/guides/installing_esphome.html][installing the ESPHome CLI tool]] and tossing some configuration together.
|
||
** Wiring up a prototype
|
||
|
||
#+caption: Clockwise from the bottom: The ESP8266 Wemos D1 mini clone wired into the breadboard, the reed switch plate, its accommpanying magnet, and the relay switch.
|
||
#+ATTR_ORG: :width 500
|
||
[[file:images/garage-door-wiring.jpg]]
|
||
|
||
|
||
To test the module out, I wired it onto a breadboard, using it's 3.3V pin to
|
||
supply power to the top positive rail and hooked its ground pin to the lower
|
||
negative rail. Hooking it up to the breadboard with only jumper wires is a bit
|
||
finicky (I didn't have a soldering iron at the time), so to confirm that
|
||
everything was working I ran wires to hook a blue LED up between the power and
|
||
ground. When everything was snug and the USB cable was plugged in, it lit up!
|
||
Success!
|
||
|
||
*** Preparing the initial firmware
|
||
I used the ESPHome CLI wizard to generate my initial firmware configuration for
|
||
the device. The wizard prompts for various values needed for basic
|
||
functionality. In my case, I specify a name for my device (=garage-door=), the
|
||
microcontroller I'm using (=ESP2866=), the board I'm using (=d1_mini=), and some
|
||
WiFi credentials. Substitute in the credentials for the WiFi network your device
|
||
will connect to if you're following along.
|
||
|
||
#+begin_center
|
||
#+caption: Video demonstration of the ESPHome CLI wizard.
|
||
#+attr_html: :width 800 :height 406 :controls t
|
||
#+begin_video
|
||
<source src="/videos/esphome-wizard.webm" type="video/webm">
|
||
Download the <a href="/videos/esphome-wizard.webm">WEBM footage</a> of the ESPHome CLI wizard.
|
||
#+end_video
|
||
#+end_center
|
||
|
||
Review the contents of the generated YAML file, then connect the board to your
|
||
computer via USB and run:
|
||
: esphome run garage-door.yml
|
||
|
||
The CLI tool will generate and compile the code for you, then prompt you for the
|
||
device to install to. As this is the first time you're installing the firmware,
|
||
you must select your USB device (in my case, on my linux machine, the device was
|
||
=/dev/ttyUSB0 (USB2.0-Ser!)=). You'll see the logs as the device boots up and
|
||
connects to your network, and it's up and running! Not doing much yet, but it is
|
||
there and discoverable!
|
||
|
||
*** Adding it to Home Assistant
|
||
Now that the device is running and discoverable on the network, it can be added
|
||
to Home Assistant. Home Assistant should detect the device on your network and
|
||
show it as a new device to add automatically in the Integrations tab of your
|
||
settings. Home Assistant will prompt for its password, which is in the =api:=
|
||
section of =garage-door.yml= (the same password that was set in the wizard). If
|
||
for some reason it doesn't, click the "+ Add Integration" button, search for and
|
||
select "ESPHome". Home Assistant then prompts for the connection settings (in my
|
||
case, the hostname was =garage-door.local=, and the default port is =6053=). As
|
||
entities are added to the ESPHome configuration and uploaded to the device, they
|
||
will become available within Home Assistant.
|
||
*** Wiring up the garage door detector
|
||
|
||
The first thing I hooked up was the reed switch. One wire is joined to the =D1=
|
||
pin on the ESP, and the other to ground. In the ESPHome configuration, I added a
|
||
binary sensor for the switch, configuring the =D1= for input with its pull-up
|
||
resistor enabled, which sets the D1 state to high normally. When the magnet is
|
||
within a couple inches of the switch plate, the switch will close the circuit,
|
||
triggering a state change from high to low on the ESP pin as the current can now
|
||
flow to the ground pin.
|
||
|
||
#+caption: Configuration YAML for the garage door sensor.
|
||
#+begin_src yaml
|
||
binary_sensor:
|
||
- platform: gpio
|
||
id: garage_door_sensor
|
||
name: "Garage Door"
|
||
device_class: garage_door
|
||
pin:
|
||
number: "D1"
|
||
mode:
|
||
input: true
|
||
pullup: true
|
||
#+end_src
|
||
|
||
The binary sensor is using the =gpio= platform to read the =D1= pin in =input=
|
||
mode with its =pullup= enabled. The =id= value will be used to reference the
|
||
sensor in other areas of the configuration, and the =device_class= is used to
|
||
inform Home Assistant that this device is monitoring a garage door.
|
||
|
||
I executed the =esphome run= command once again to load the new firmware, and
|
||
once it was up and running I was able to verify that the switch was working from
|
||
the logs as I moved the magnet up to and away from the switch plate.
|
||
|
||
#+caption: The ESPHome logs showing the door state updating.
|
||
#+ATTR_ORG: :width 500
|
||
[[file:images/garage-door-sensor.png]]
|
||
|
||
I was also able to add the new sensor entity to my Home Assistant dashboard,
|
||
which also updated as I moved the magnet!
|
||
|
||
#+caption: The garage door sensor card in Home Assistant.
|
||
[[file:images/garage-door-sensor-card.png]]
|
||
|
||
*** Adding the garage door control switch
|
||
Next is the relay switch, which I will use to control the garage door so I can
|
||
open or close it via the Home Assistant mobile app or any automations I decide
|
||
to set up for it.
|
||
|
||
In the ESPHome configuration, I added a =gpio= switch using the =D5= pin. Since
|
||
going to be activating this switch in a particular way within another control, I
|
||
marked it as =internal= so it can't be activated manually via Home Assistant. I
|
||
then added a =cover= entity representing the garage door as a whole. This wraps
|
||
up the door sensor and door control into one neat package, and lets me specify
|
||
how the switch should be toggled to activate the door.
|
||
|
||
#+caption: Configuration YAML for the opener control switch.
|
||
#+begin_src yaml
|
||
switch:
|
||
- platform: gpio
|
||
id: garage_door_switch
|
||
name: "Garage Door Opener"
|
||
pin: "D5"
|
||
internal: true
|
||
|
||
cover:
|
||
- platform: template
|
||
name: "Garage Door"
|
||
lambda: |-
|
||
if (id(garage_door_sensor).state) {
|
||
return COVER_OPEN;
|
||
} else {
|
||
return COVER_CLOSED;
|
||
}
|
||
open_action:
|
||
- switch.turn_on: garage_door_switch
|
||
- delay: 0.1s
|
||
- switch.turn_off: garage_door_switch
|
||
close_action:
|
||
- switch.turn_on: garage_door_switch
|
||
- delay: 0.1s
|
||
- switch.turn_off: garage_door_switch
|
||
stop_action:
|
||
- switch.turn_on: garage_door_switch
|
||
- delay: 0.1s
|
||
- switch.turn_off: garage_door_switch
|
||
#+end_src
|
||
|
||
Because the garage has only one switch for opening, closing, and stopping the
|
||
door, the =open_action=, =close_action=, and =stop_action= are identical. To
|
||
trigger the mechanism, it activates the switch, pauses briefly, then deactivates
|
||
the switch. I used a =lambda= to interrogate the door sensor's state to return
|
||
whether the "cover" is open or closed.
|
||
|
||
The switch uses three hookups to the ESP: One from the =3.3v= pin, one from the
|
||
input wired to the =D5= pin, and one to ground. To give it something to control
|
||
while testing in place of the garage door opener it'll eventually connect to, I
|
||
set up a circuit with a red LED between the 3.3v power and ground lines, and
|
||
wired the relay switch in the middle. Because I only want the LED (door opener)
|
||
circuit closed when the switch is activated (i.e. a /normally-open circuit/), I
|
||
attached the LED circuit wires to the =NO= (Normally Open) and =COM= (Common)
|
||
leads on the far side of the switch.
|
||
|
||
Once this was done, I was able to activate the door control in Home Assistant
|
||
and see the red LED toggled on and off!
|
||
|
||
By adding the cover entity to Home Assistant, I was also able to get this nifty control card!
|
||
|
||
#+CAPTION: The garage door control card in Home Assistant.
|
||
[[file:images/garage-door-cover-card.png]]
|
||
|
||
*** Celebration!
|
||
Behold! The prototype works!
|
||
|
||
#+begin_center
|
||
#+caption: Video demonstration of the functioning prototype.
|
||
#+attr_html: :width 480 :height 960 :controls t
|
||
#+begin_video
|
||
<source src="/videos/garage-door-prototype.webm" type="video/webm">
|
||
Download the <a href="/videos/garage-door-prototype.webm">WEBM footage</a> of the wired prototype.
|
||
#+end_video
|
||
#+end_center
|
||
|
||
** Hooking it all up for real
|
||
With the prototype sorted, it was time to put it together into something I could
|
||
install!
|
||
|
||
*** Getting it soldered
|
||
I grabbed a perforated board and got to wiring and soldering everything
|
||
together, changing the pins I was using for the sensor and controller to make
|
||
them easier to route (now =D5= and =D2=, respectively). For my first time
|
||
soldering electronics, it went pretty smoothly. Getting the wires soldered
|
||
together took a couple of tries and aren't going to win any beauty contests, but
|
||
everything's secure and working fine.
|
||
|
||
I also added a red status LED adjacent to the blue one I used for power that
|
||
will flash when booting or in an error state.
|
||
|
||
#+caption: Configuration YAML for the status LED.
|
||
#+begin_src yaml
|
||
status_led:
|
||
pin: "D7"
|
||
#+end_src
|
||
|
||
#+caption: The top, all neatly connected to the board.
|
||
[[file:images/garage-door-soldered-top.png]]
|
||
#+caption: The bottom, all wired up.
|
||
[[file:images/garage-door-soldered-bottom.png]]
|
||
|
||
*** Designing and printing an enclosure
|
||
With everything in place, I grabbed my calipers, measured everything, and hopped
|
||
into FreeCAD. I put in some supports to hold the board up with some room
|
||
underneath for the wires, and cut out holes in the sides where the usb power and
|
||
external wires will attach to their screw mounts. I added small bumps on
|
||
opposite inside walls to grab the edges of the board and hold it in place. I
|
||
then made a lid that will slide in to the top, and opened holes in it for the
|
||
LED to shine through and to provide access to the screws for the wire mounts.
|
||
The lid also has a small bump for it to hold it in place when fully inserted.
|
||
|
||
#+caption: The completed design in FreeCAD.
|
||
[[file:images/garage-door-freecad.png]]
|
||
|
||
There are pretty visible layer lines in the print, likely because I switched in
|
||
a new roll of filament for this and didn't dial it in first, but where this
|
||
thing is going it doesn't need to be gorgeous.
|
||
|
||
#+caption: The case and lid freshly printed in white PLA.
|
||
[[file:images/garage-door-case.png]]
|
||
|
||
The board popped right into place, nice and cozy in its new home. The openings
|
||
lined up pretty well, though I did have to widen the USB opening a bit with an
|
||
X-Acto knife so the cable would fit properly.
|
||
|
||
#+caption: Snug in its new home!
|
||
[[file:images/garage-door-in-case.png]]
|
||
|
||
The lid slid right in and locked in place just fine! The LEDs shine through
|
||
their opening, and the screws are easy to get to and manipulate with a
|
||
screwdriver.
|
||
|
||
#+caption: All tucked in with its lid!
|
||
[[file:images/garage-door-with-lid.png]]
|
||
|
||
I'm proud of my little creation and its pretty little case. Next up, installation!
|
||
*** Installation
|
||
Alright, time to get this thing into the garage! Getting this set up was pretty
|
||
straightforward. I decided to install the reed switch sensor at the top of the
|
||
garage door, screwing the sensor into wood above it and bolting the magnet onto
|
||
the top of the metal door such that the magnet is positioned beside the sensor
|
||
when the door is closed. Using my laptop, I was able to monitor the device and
|
||
see that the switch did correctly register the door's state.
|
||
|
||
#+caption: The switch is mounted to the wall with its wires running up to the rail, the magnet is mounted to the top of the door.
|
||
[[file:images/garage-door-installation-sensor.png]]
|
||
|
||
To connect everything together, I cut some lengths of [[https://www.amazon.com/gp/product/B076645YB7][bell wire]] to the distances
|
||
I needed, and got started. I ran a pair of wires from the switch from there to
|
||
the garage door opener, where I mounted the device to its frame with some ties.
|
||
I may attach it to the ceiling later so it's prettier, but for now this works
|
||
just fine. I then wired the =NO= and =COM= connectors on the relay switch to the
|
||
two leftmost connectors in my garage door opener, which are the two connectors
|
||
shared with the wall garage door button.
|
||
|
||
#+caption: The wires were added to the garage door opener alongside the wall switch's existing wires.
|
||
[[file:images/garage-door-installation-opener.png]]
|
||
|
||
#+caption: The device is mounted to the metal frame with ties, having drilled a couple holes into the back of the enclosure to loop them through.
|
||
[[file:images/garage-door-installation-device.png]]
|
||
|
||
With everything connected, I powered up my device with a USB power brick plugged
|
||
into the outlet above the door opener, and... it worked! I was able to open and
|
||
close the door using Home Assistant on my laptop or on my phone, and get
|
||
feedback on whether the door was left open or closed!
|
||
** Wrapping up
|
||
This was a really cool project! I'm super proud of it and /very/ happy with the
|
||
result, and I learned a great deal about building electronics along the way. I'm
|
||
looking forward to finding more ways to make our home just a little bit smarter
|
||
and easier for us to manage, and I expect I'll have plenty of fun putting
|
||
together even more electronic projects in the future!
|
||
* DONE On Twitter's fall and the rise of the Fediverse
|
||
CLOSED: [2022-11-18 Fri 12:47]
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: on-twitter-and-the-fediverse
|
||
:EXPORT_DATE: 2022-11-18
|
||
:END:
|
||
|
||
#+begin_quote
|
||
tl;dr: Mastodon is not Twitter, and that's great for many reasons, but I have no
|
||
idea what, if anything, will end up filling the giant void that'll be left when
|
||
the birdsite disintegrates.
|
||
#+end_quote
|
||
|
||
I've found a [[https://tech.lgbt/@correlr][happy new home in the Fediverse]], but I find myself wondering what
|
||
the broader effects of Twitter's implosion will be. Navigating Mastodon's
|
||
federated nature is a stumbling block for lots of people. I do wonder what will
|
||
come of it next, but I am liking how much interest there genuinely is for
|
||
networks not controlled by a single entity.
|
||
|
||
A federated network really can't compare with the reach afforded by an
|
||
entrenched, centralized platform. It's a lot harder to get in front or even find
|
||
a lot of diverse new people without platform-wide virality and algorithmic
|
||
gaming. It's fundamentally a different type of network, and folks who are
|
||
reliant on it for their audiences aren't going to have a good time with it going
|
||
away. The low friction of a centralized network for people to join and content
|
||
to reach them just can't be beat.
|
||
|
||
Things may change if the Fediverse reaches a critical mass, but I don't see that
|
||
happening (at least not anytime soon). Twitter's fall is going to leave quite a
|
||
void to be filled, and I'm not sure what will end up claiming it. Worst case,
|
||
nothing does for a long time, and a lot of social organization is going to
|
||
struggle with being siloed away for a good while.
|
||
|
||
Maybe Twitter will somehow recover, but it's hard to imagine it will without its
|
||
staff. I'm wondering if it'll end up remaining as a company, but be forced to
|
||
pivot to different software as the current platform degrades with lack of
|
||
maintenance and experts to guide new development. Given it's scale, though, I
|
||
don't really think that's a tenable option either. What's it going to do,
|
||
attempt to ETL everything into an unfederated mastodon fork?
|
||
|
||
I wouldn't have known how to find other trans people without Twitter. Maybe I'd
|
||
have eventually found some weird FB groups (or worse, reddit), but none of the
|
||
other options are built for those communities to find their ways in front of you
|
||
without deliberately seeking them out.
|
||
|
||
So, yeah, regardless I'm most worried about the social impact all of this will
|
||
have. Twitter was pretty instrumental to a lot of recent cultural awareness,
|
||
uprisings, unionization efforts, and other such things. If it does collapse, I'm
|
||
not sure what'll come of it. It's a trash-filled hellsite for sure, but it's
|
||
also been an incredibly powerful tool.
|
||
* TODO Turning up the volume :micropython:python:electronics:3d_printing:programming:
|
||
:PROPERTIES:
|
||
:EXPORT_FILE_NAME: turning-up-the-volume
|
||
:EXPORT_DATE: 2023-06-21
|
||
:END:
|
||
|
||
I've had these speakers for over twenty years, and for most of them the volume
|
||
selector has been awful. The speakers themselves are fine, a simple enough pair
|
||
of small stereo speakers with an accompanying subwoofer, but the volume selector
|
||
is a plastic wheel set in-line with the speaker cable which — when rotated to all
|
||
but a few narrow positions — fails to make a solid connection and leaves at least
|
||
one of the two speakers cut out. I've put up with it for quite a while, the
|
||
positions being good enough and stable enough to enjoy it, but now that I've got
|
||
a proper electronics hobby and am far less shy about cutting wires apart I'm
|
||
found myself glancing at it thinking it'd make a great new project.
|
||
|
||
|
||
Cutting the wires and unscrewing the case, the dial was revealed to be a very
|
||
simple 10kΩ potentiometer. While this could /easily/ be replaced with a newer,
|
||
prettier knob, there's just no fun in so simple a project. Indeed, why not have
|
||
it controllable via WiFi? Or select multiple inputs? I'm using it primarily for
|
||
my turntable right now, but I've got a TV nearby, and it'd certainly be nice to
|
||
pipe my computer's audio through it when I feel like it.
|