Upgrade to Pro — share decks privately, control downloads, hide ads and more …

RailsConf 2017

RailsConf 2017

It's my keynote from RailsConf 2017

Aaron Patterson

May 05, 2017
Tweet

More Decks by Aaron Patterson

Other Decks in Programming

Transcript

  1. "We are, temporarily, in a kind of dark ages of

    end-user open source software."
  2. "No generics and no sum types? Those are no longer

    groundbreaking, they're the bare minimum.*" *Emphasis not mine
  3. 5

  4. 5

  5. $5

  6. 5.1

  7. Some things I don’t have time to do but you

    should do them for me Maybe like 3 things?
  8. HTTP/2 (really) • Binary protocol • Multiplexes (multiple requests on

    the same socket) • Secure (only works over SSL*) • Push responses
  9. Link Header 200 OK Content-Length: 1024 Content-Type: text/html Link: </choochoo.css>;

    rel=preload Link: </gorby.js>; rel=preload response body
  10. Server (Unicorn) Proxy (h2o) Client HTTP/2 HTTP/1.1 200 OK Content-Length:

    1024 Content-Type: text/html Link: </choochoo.css>; rel=preload Link: </gorby.js>; rel=preload response body
  11. Link Header Problem 200 OK Content-Length: 1024 Content-Type: text/html Link:

    </choochoo.css>; rel=preload Link: </gorby.js>; rel=preload response body
  12. 103 Early Response 103 Early Hints Link: </choochoo.css>; rel=preload Link:

    </gorby.js>; rel=preload 200 OK Content-Length: 1024 Content-Type: text/html response body
  13. Data I acquired (crap data) -100 925 1950 2975 4000

    Time 2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32
  14. Implementation instructions = [ [:push, 3], [:push, 7], [:add], ]

    pc = 0 loop do instruction = instructions[pc] pc += 1 send function_table[instruction.first] end
  15. Loop Elimination instructions = [ [:push, 3], [:push, 7], [:add],

    ] def push_op pc, val, instructions # code to push on the stack # .... next_instruction = instructions[pc] send(function_table[next_instruction.first] pc + 1, next_instruction, instructions) end def add pc, val, instructions # code to add # .... next_instruction = instructions[pc] send(function_table[next_instruction.first] pc + 1, next_instruction, instructions) end # start of VM pc = 0 instruction = instructions[pc] pc += 1 send function_table[instruction.first] Generated
  16. Function Name Lookup Table instructions = [ [:push, 3], [:push,

    7], [:add], ] function_table = { :push => :push_op, :add => :add_op, } def push_op pc, val, instructions # code to push on the stack # .... next_instruction = instructions[pc] send(function_table[next_instruction.first] pc + 1, next_instruction, instructions) end
  17. JUMP instructions = [ [:push, 3], [:push, 7], [:add], ]

    function_table = { :push => ADDRESS_OF(:push_op), :add => ADDRESS_OF(:add_op), } label: :add_op # code to add # ... next_instruction = instructions[pc] val = next_instruction.drop(1) address = function_table[next_instruction.first] pc += 1 GOTO(address) label: :push_op # code to push on the stack # .... next_instruction = instructions[pc] val = next_instruction.drop(1) address = function_table[next_instruction.first] pc += 1 GOTO(address) Generated
  18. Addresses in Byte Code instructions = [ [ADDRESS_OF(:push_op), 3], [ADDRESS_OF(:push_op),

    7], [ADDRESS_OF(:add)], ] label: :add_op # code to add # ... next_instruction = instructions[pc] val = next_instruction.drop(1) address = next_instruction.first pc += 1 GOTO(address) label: :push_op # code to push on the stack # .... next_instruction = instructions[pc] val = next_instruction.drop(1) address = next_instruction.first pc += 1 GOTO(address)
  19. Extremely Simple Code require 'lolvm' class MyRuby attr_reader :rb_path2class, :rb_define_method,

    :mod def initialize ctx @mod = ctx.create_module 'ruby' @ctx = ctx init_functions end private def init_functions @rb_path2class = make_func('rb_path2class', @ctx.int8.pointer, [@ctx.int8.pointer]) @rb_define_method = make_func('rb_define_method', @ctx.int8.pointer, [@ctx.int8.pointer, @ctx.int8.pointer, @ctx.int8.pointer, @ctx.int32]) end def make_func name, ret, args func_type = LOLVM::Types::Function.new(ret, args, false) @mod.add_function name, func_type end end context = LOLVM::Context.instance builder = context.create_builder ruby = MyRuby.new context mod = context.create_module 'anon mod' func_type = LOLVM::Types::Function.new(context.double, [], false) testing = mod.add_function "testing", LOLVM::Types::Function.new(context.int64, [context.int8.pointer], false) testing.param(0).name = "foo" bb = context.append_basic_block testing, "entry" x = mod.add_global context.int64, 'rb_cObject' x.linkage = LOLVM::Linkages::External builder.position_at_end_of bb builder.ret builder.load x, "obj" func = mod.add_function "__anon", func_type bb = context.append_basic_block func, "entry" builder.position_at_end_of bb str = builder.global_string "MyRuby", 'string' func_name = builder.global_string "testing", 'func_name' my_ruby = builder.call testing, [str], 'rb_str' builder.ret my_ruby ee = mod.execution_engine ee.add_module ruby.mod p :GOT => ee.run("__anon") Assembles machine code at run time using LLVM
  20. Data I acquired (crap data) -100 925 1950 2975 4000

    Time 2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32
  21. Discrete Signal (crap data) -100 925 1950 2975 4000 Time

    2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32
  22. Signal Convolution -100 925 1950 2975 4000 Time 2017-01-23 15:53:51

    2017-01-23 15:54:01 2017-01-23 15:54:11 2017-01-23 15:54:21 2017-01-23 15:54:31 2017-01-23 15:54:42 2017-01-23 15:54:52 2017-01-23 15:55:02 2017-01-23 15:55:12 2017-01-23 15:55:22 2017-01-23 15:55:32 0 0.25 0.5 0.75 1 0 1 2 3 4 5 6 7 * =?
  23. X -1.5 -0.5 0.5 1.5 0 1 2 3 4

    5 6 7 8 H -0.5 0 0.5 1 3 2 1 0 Y -1.5 0 1.5 3 0 1 2 3 4 5 6 7 8 9 10 11
  24. Convolution Code def convolve x, h Array.new(x.length + h.length -

    1) do |i| h.map.with_index { |h_j, j| if i - j < 0 || i - j > x.length - 1 0 else h_j * x[i - j] end }.inject(:+) end end
  25. X -1.5 -0.5 0.5 1.5 0 1 2 3 4

    5 6 7 8 H -0.5 0 0.5 1 3 2 1 0 x + y[i] value
  26. Identity Signal X -1.5 -0.5 0.5 1.5 0 1 2

    3 4 5 6 7 8 H 0 0.25 0.5 0.75 1 0 1 2 3 * = Y -1.5 -0.5 0.5 1.5 0 1 2 3 4 5 6 7 8 9 10 11
  27. Amplifier X -1.5 -0.5 0.5 1.5 0 1 2 3

    4 5 6 7 8 H 0 0.5 1 1.5 2 0 1 2 3 * = Y -3 -1 1 3 0 2 4 6 8 10
  28. Shift X -1.5 -0.5 0.5 1.5 0 1 2 3

    4 5 6 7 8 H 0 0.25 0.5 0.75 1 0 1 2 3 * = Y -1.5 -0.5 0.5 1.5 0 1 2 3 4 5 6 7 8 9 10 11
  29. Echo X -1.5 -0.5 0.5 1.5 0 1 2 3

    4 5 6 7 8 H 0 0.5 1 1.5 2 0 1 2 3 * = Y -3.75 -1.25 1.25 3.75 0 1 2 3 4 5 6 7 8 9 10 11
  30. Average X -1.5 -0.5 0.5 1.5 0 1 2 3

    4 5 6 7 8 H 0 0.25 0.5 0.75 1 0 1 2 3 * H 0 0.125 0.25 0.375 0.5 0 1 2 3 *
  31. Convolution Code def convolve x, h Array.new(x.length + h.length -

    1) do |i| h.map.with_index { |h_j, j| if i - j < 0 || i - j > x.length - 1 0 else h_j * x[i - j] end }.inject(:+) end end
  32. Arrays are 1 based > data <- c(4, 3, 2,

    1) > data[0] numeric(0) > data[0] + 1 numeric(0) >
  33. Data I acquired (crap data) -100 925 1950 2975 4000

    Time 2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32
  34. cons tenderlove.cons <- function(x, y) { function(m) { m(x, y)

    } } R Node cons = ->(x, y) { ->(m) { m.(x, y) } } Ruby var cons = function(x, y) { return function(m) { return m(x, y); }; }
  35. car tenderlove.car <- function(z) { z(function(p, q) { p })

    } R var car = function(z) { return z(function(p, q) { return p; }); } Node car = ->(z) { z.(->(p, q) { p }) } Ruby
  36. Usage > tenderlove.cons(1, 2) function(m) { m(x, y) } <environment:

    0x7f8c55961f80> > tenderlove.car function(z) { z(function(p, q) { p }) } > tenderlove.car(tenderlove.cons(1, 2)) [1] 1 >
  37. cdr tenderlove.cdr <- function(z) { z(function(p, q) { q })

    } R var cdr = function(z) { return z(function(p, q) { return q; }); } Node cdr = ->(z) { z.(->(p, q) { q }) } Ruby
  38. each tenderlove.each <- function(func, lst) { if (!is.null(lst)) { element

    <- tenderlove.car(lst) func(element) tenderlove.each(func, tenderlove.cdr(lst)) } } R Node each = ->(func, list) { if list element = car.(list) func.(element) each.(func, cdr.(list)) end } Ruby var each = function(func, list) { if(list) { element = car(list); func(element); each(func, cdr(list)); } }
  39. Usage var each = function(func, list) { if(list) { element

    = car(list); func(element); each(func, cdr(list)); } } each(console.log, cons(1, (cons(2, cons(3, null))))); [aaron@TC dsp (master)]$ node dsp.js 1 2 3 [aaron@TC dsp (master)]$
  40. Numbers tenderlove.zero <- function(f) { function(x) { x } }

    tenderlove.one <- function(f) { function(x) { f(x) } } tenderlove.two <- function(f) { function(x) { f(f(x)) } } tenderlove.three <- function(f) { function(x) { f(f(f(x))) } } print(tenderlove.three(function(x) { x + 1 })(0)) => 3
  41. Addition tenderlove.one <- function(f) { function(x) { f(x) } }

    tenderlove.add <- function(a, b) { function(f) { function(x) { a(f)(b(f)(x)) } } } tenderlove.two <- tenderlove.add(tenderlove.one, tenderlove.one) tenderlove.three <- tenderlove.add(tenderlove.one, tenderlove.two) print(tenderlove.three(function(x) { x + 1 })(0)) # => 3
  42. true / false / if True = ->(x, y) {

    x } False = ->(x, y) { y } If = ->(cond, t, f) { cond.(t, f) } If.(True, ->() { :true }, ->() { :false }).call # => :true If.(False, ->() { :true }, ->() { :false }).call # => :false
  43. My Signal (crap data) -100 925 1950 2975 4000 Time

    2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32
  44. cons / car / cdr / each tenderlove.cons <- function(x,

    y) { function(m) { m(x, y) } } tenderlove.car <- function(z) { z(function(p, q) { p }) } tenderlove.cdr <- function(z) { z(function(p, q) { q }) } tenderlove.each <- function(func, lst) { if (!is.null(lst)) { element <- tenderlove.car(lst) func(element) tenderlove.each(func, tenderlove.cdr(lst)) } } tenderlove.each_with_index <- function(func, lst) { idx <- 0 tenderlove.each(function(element) { func(idx, element) idx = idx + 1 }, lst) }
  45. array indexing tenderlove.vec_at <- function(idx, lst) { if(idx == 0)

    { tenderlove.car(lst) } else { tenderlove.vec_at(idx - 1, tenderlove.cdr(lst)) } }
  46. Converting Vectors to cons tenderlove.c2cons <- function(cs) { foldl <-

    function(proc, init, lst) { if(length(lst) == 0) { init } else { foldl(proc, proc(head(lst, 1), init), tail(lst, -1)) } } foldl(tenderlove.cons, NULL, rev(cs)) }
  47. Convolution in R tenderlove.conv <- function(x, h) { x_list <-

    tenderlove.c2cons(x) h_list <- tenderlove.c2cons(h) indexes <- 0:(length(x) + length(h) - 1) sapply(indexes, function(i) { sum(tenderlove.cons2c(tenderlove.map_with_index(function(j, h_j) { if (i - j < 0 || i - j > length(x) - 1) { 0 } else { h_j * tenderlove.vec_at(i - j, x_list) } }, h_list))) }) }
  48. Feed in test data > data <- c(0, -1, -1.2,

    2, 1.5, 1.4, 0.5, 0, -0.5) > plot(data) > filter <- c(2, 0, 1) > plot(filter) > plot(tenderlove.conv(data, filter))
  49. Input Real Data > plot(tenderlove.conv(unlist(dat$V2), c(1))) Error: evaluation nested too

    deeply: infinite recursion / options(expressions=)? I f#$@*ing hate R. tenderlove
  50. My Signal (crap data) -100 925 1950 2975 4000 Time

    2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32
  51. I made this Weight (g) -100 925 1950 2975 4000

    Time 2017-01-23 15:53:51 2017-01-23 15:53:56 2017-01-23 15:54:01 2017-01-23 15:54:06 2017-01-23 15:54:11 2017-01-23 15:54:16 2017-01-23 15:54:21 2017-01-23 15:54:26 2017-01-23 15:54:31 2017-01-23 15:54:37 2017-01-23 15:54:42 2017-01-23 15:54:47 2017-01-23 15:54:52 2017-01-23 15:54:57 2017-01-23 15:55:02 2017-01-23 15:55:07 2017-01-23 15:55:12 2017-01-23 15:55:17 2017-01-23 15:55:22 2017-01-23 15:55:27 2017-01-23 15:55:32 Cat Enters Cat Leaves