This commit is contained in:
Anton Dyudin 2014-08-11 20:14:00 -07:00 committed by Anton Dyudin
commit 0556777508
68 changed files with 9161 additions and 3336 deletions

View File

@ -439,7 +439,7 @@
vix=(bex +((cut 0 [25 2] mag))) :: width of sender
tay=(cut 0 [27 5] mag) :: message type
==
?> =(5 vez)
?> =(0 vez)
?> =(chk (end 0 20 (mug bod)))
:+ [(end 3 wix bod) (cut 3 [wix vix] bod)]
(kins tay)
@ -459,7 +459,7 @@
=+ tay=(ksin q.kec)
%+ mix
%+ can 0
:~ [3 5]
:~ [3 0]
[20 (mug bod)]
[2 yax]
[2 qax]
@ -1047,7 +1047,7 @@
++ gnaw :: gnaw:am
|= [kay=cape ryn=lane pac=rock] :: process packet
^- [p=(list boon) q=fort]
?. =(5 (end 0 3 pac)) [~ fox]
?. =(0 (end 0 3 pac)) [~ fox]
=+ kec=(bite pac)
?: (goop p.p.kec) [~ fox]
?. (~(has by urb.ton.fox) q.p.kec)

View File

@ -205,7 +205,7 @@
|= [orx=oryx moh=moth]
^- (unit ,[hasp path])
=+ jun=(ecci orx moh)
~& [%ecca jun]
:: ~& [%ecca jun]
?~ jun ~
=+ ^- (unit ,[his=term app=term pax=term])
%. u.jun
@ -236,9 +236,12 @@
=+ jun=(ecce moh)
?~ jun ~
?. ?=(%o -.u.jun) ~
?. =([~ %s orx] (~(get by p.u.jun) %oryx))
~& [%ecci-oryx u.jun]
~
:: ?. =([~ %s orx] (~((get by p.u.jun) %oryx))
:: ~& [%oryx-sent ~(get by p.u.jun) %oryx)]
:: ~& [%oryx-good orx]
:: ~
~? !=([~ %s orx] (~(get by p.u.jun) %oryx))
[%oryx [%sent (~(get by p.u.jun) %oryx)] [%good orx]]
=+ nuj=(~(get by p.u.jun) %xyro)
?~(nuj [~ ~] [~ u.nuj])
::
@ -387,8 +390,9 @@
(slav %ud i.t.t.t.tea)
?~ ouy
+>.$
?: (lth ~m2 (sub now tim.bet.siq:beat:u.ouy))
abet:work:amok:u.ouy
:: ?: (lth ~m2 (sub now tim.bet.siq:beat:u.ouy))
:: ~& %axon-heartbeat
:: abet:work:amok:u.ouy
=* mab t.t.t.t.tea
=+ woy=(yule:u.ouy ?+(i.mab !! %mess %meg, %show %sub))
=< abet =< work =< abet
@ -437,7 +441,8 @@
yoo(can.sub.siq (~(put by can.sub.siq.yoo) nap sem(num ~)))
==
%nice
?. ?=(%mess i.mab) u.ouy
?. ?=(%mess i.mab)
u.ouy
(hear:woy ~ %& %json !>((joba %ok %b &)))
?(%rust %rush)
?< ?=(~ t.mab)
@ -1190,10 +1195,20 @@
if(cb) {
xhr.onload = function() {
cb(null,{
status:this.status,
data:JSON.parse(this.responseText)
})
try {
err = null
res = {
status:this.status,
data: JSON.parse(this.responseText)
}
} catch(e) {
err = {
message:"Failed to parse JSON",
raw:this.responseText
}
res = null
}
cb(err,res)
}
xhr.onerror = function() {
cb({

View File

@ -629,7 +629,7 @@
|= a=(pair ship desk)
:- hun.mat
:^ %pass (away %w %drug (scot %p p.a) q.a ~) %c
[%warp [our p.a] q.a ~ %| [%da now] [%da (add now ~d1000)]]
[%warp [our p.a] q.a ~ %| [%da +(now)] [%da (add now ~d1000)]]
=+ ^= old ^- (list move)
%+ turn
%+ skip (~(tap in ped.sat) ~)

View File

@ -809,7 +809,11 @@
::
++ jesc
|= a=@ ^- tape
?.(=(10 a) [a ~] "\\n")
?+ a [a ~]
10 "\\n"
34 "\\\""
92 "\\\\"
==
::
++ taco :: atom to octstream
|= tam=@ ^- octs

114
main/app/a-twit/core.hook Normal file
View File

@ -0,0 +1,114 @@
/+ twitter
!:
=> |%
++ rev 0
++ axle ,[@ @]
++ gilt :: State data
$% [%json p=json] :: JSON format
[%html p=@t]
[%hymn p=manx] :: HTML format
==
++ gift :: Message data
$% [%rust gilt] :: Refresh webapp
[%rush gilt] :: Update webapp
[%mean (unit (pair term (list tank)))] :: Error, maybe w/ msg
[%nice ~] :: Response message
==
++ move ,[p=bone q=(mold note gift)] :: Arvo command
++ sign
$% $: %e
$% [%thou p=httr]
== == ==
++ note :: Arvo message
$% $: %e
$% [%them p=(unit hiss)]
== == ==
++ app-key :: specific keys
:- :- 'hDDOTPfGHGlsOUbhpy6qc6XbW'
'olCkea6wm3XG4pnVCHuPIozUF2ggH1sHjnBtuT4Ai6rCOeQGzO'
:- '2485712317-R77Lpdu5rAJadRVxTXPpnxvcwS0IfNG7QEzLPty'
'a41d83XId0P7QQbodkPYv3zxoEL0Cq9EsN2eXZBZAwAWA'
--
|_ [hid=hide vat=axle]
::
++ root :: App root location
/(scot %p our.hid)/main/(scot %da lat.hid)/app/[app.hid]
::
++ incl :: Include scripts
|= wal=wall
%+ turn wal
|= tape ;script(type "text/javascript", src +<);
::
++ peer :: Accept subscribes
|= [ost=bone you=ship pax=path]
^- [(list move) _+>]
?~ pax
:: [[ost %give %rust %hymn page]~ +>.$]
[[ost %pass /curl1 %e [%them (some timl)]]~ +>.$]
[~ +>.$]
::
++ poke-json
|= [ost=bone his=ship jon=json]
^- [(list move) _+>]
:: this is where you take the jon and post it to twitter.
~& [%got-json jon]
[[ost %pass /curl2 %e [%them (some (posl jon))]]~ +>.$]
::[[ost %give %nice ~]~ +>.$]
::
++ page
|= a=@t
^- manx
;html
;head
;style:"{(trip ;;(,@ .^(%cx (welp root /main/css))))}"
;* %- incl :~
"//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.1/jquery.min.js"
==
;title: Urbit - Twitter Test
==
;body
;div#c
;div#d
;div#twet
;div.author: @urbit_test
;div.date;
;textarea#tweetr(placeholder "What would you like to tweet?");
;input#submit(type "button", value "Send");
;input#length(type "button", value "0/140");
==
;div#time;
==
==
:: this is where you put the json from the home timeline in
:: make sure its in a script tag with the id jime
;script#jime:"{(trip a)}"
;script:"{(trip ;;(,@ .^(cx/(welp (scag 3 `path`root) /lib/urb/js))))}"
;script:"{(trip ;;(,@ .^(%cx (welp root /main/js))))}"
==
==
++ pour
|= [way=path sih=sign]
^- [(list move) _+>]
:_ +>.$
%+ turn (~(tap in (~(get ju pus.hid) ~)))
|= ost=bone
[ost %give %rust %hymn (page `@t`q:(need r.p.sih))]
++ testt
^- hiss
[(scan "http://www.example.com" auri:epur) %get ~ ~]
++ timl
^- hiss
=+ (twit `keys`app-key lat.hid `@`eny.hid)
(stat-home ~ ~)
++ posl
|= jon=json
=+ txt=`(unit ,[%tweet p=cord])`((of:jo [%tweet so:jo] ~) jon)
~& [%posl p:(need txt)]
^- hiss
=+ (twit `keys`app-key lat.hid `@`eny.hid)
(stat-upda ~[(st ~ p:(need txt))] ~)
--

View File

@ -6,13 +6,14 @@ input {
}
body {
margin-top: 2rem;
font-size 18px;
margin-top: 4rem;
font-size: 18px;
}
input,
textarea {
font-size: 1rem;
outline: none;
}
#c {
@ -23,8 +24,9 @@ textarea {
#d,
#tweet,
.tweet,
.text {
width: 24rem;
.text,
textarea {
width: 32rem;
}
#d {
margin-left: auto;
@ -33,14 +35,11 @@ textarea {
textarea {
border: 0;
height: 6rem;
height: 8rem;
line-height: 1.5rem;
margin-left: -1rem;
margin-bottom: .3rem;
width: 26rem;
outline: none;
resize: none;
padding: 1rem;
padding: 1rem 1px;
background-color: #f7f7f7;
}
@ -58,26 +57,61 @@ textarea:focus {
#length {
width 1rem;
height 1rem;
font-size: .6rem;
color: #ccc;
background-color: #eee;
background-color: transparent;
}
#submit {
background-color: #ccc;
background-color: transparent;
border: 2px solid #5DE668;
color: #5DE668;
padding: .3rem 1rem;
font-weight: 500;
cursor: pointer;
}
#submit:hover,
#submit:focus {
background-color: #5DE668;
color: #fff;
}
#submit.disabled {
opacity: .6;
}
.text {
word-wrap:break-word;
margin-bottom: .3rem;
line-height: 1.6rem;
}
#twet {
margin-bottom: 3rem;
border-bottom: 2px solid #464646;
padding-bottom: 2rem;
}
#twet,
.tweet {
margin-bottom: 1rem;
border-bottom: 1px solid #eee;
}
.tweet {
margin-bottom: 2rem;
padding-bottom: 1rem;
}
.author,
.date {
font-size: .8rem;
color: #ccc;
margin-right: .6rem;
display: inline-block;
margin-bottom: .3rem;
}
.author {
color: #888;
}

View File

@ -1,8 +1,8 @@
$(function() {
checkLength = function() {
if($tweet.val().length > 140) {
e.stopPropagation()
e.preventDefault()
short = $tweet.val().slice(0,140)
$tweet.val(short)
return false
}
}
@ -10,8 +10,20 @@ $(function() {
$("#length").val($("#tweetr").val().length+"/140")
}
twoDig = function(d) {
return (d<10) ? "0"+d : d
}
setTime = function() {
d = new Date()
datestr = twoDig(d.getMonth()+1) + "-" + twoDig(d.getDate()) + "-" + d.getFullYear() + " " + twoDig(d.getHours()) + ":" + twoDig(d.getMinutes()) + ":" + twoDig(d.getSeconds())
$("#twet .date").text(datestr)
}
setInterval(setTime,1000)
setTime()
$tweet = $("#tweetr")
$time = $("#time")
$submit = $('#submit')
$tweet.focus()
$tweet[0].selectionStart = $tweet[0].selectionEnd = $tweet.val().length
@ -20,15 +32,20 @@ $(function() {
$tweet.keydown(checkLength)
$tweet.keyup(setLength)
$('#submit').click(function() {
$submit.click(function() {
tweet = $tweet.val()
$tweet.attr('disabled', true)
$submit.attr('disabled', true)
$submit.addClass('disabled')
window.urb.send({
appl:"twit",
data:{tweet:tweet}
}, function(err,res) {
console.log(arguments)
$tweet.attr('disabled', true)
$tweet.attr('disabled', false)
$submit.attr('disabled', false)
$submit.removeClass('disabled')
$tweet.val('')
})
})
@ -39,8 +56,9 @@ $(function() {
d = new Date(tweets.created_at)
datestr = d.getMonth()+1 + "-" + d.getDate() + "-" + d.getFullYear() + " " + d.getHours() + ":" + d.getMinutes() + ":" + d.getSeconds()
$tweets = $("<div class='tweet'></div>")
$tweets.append("<div class='text'>"+tweets.text+"</div>")
$tweets.append("<div class='author'>@urbit_test</div>")
$tweets.append("<div class='date'>"+datestr+"</div>")
$tweets.append("<div class='text'>"+tweets.text+"</div>")
$time.append($tweets)
}
})

File diff suppressed because one or more lines are too long

117
main/app/twit/app.js Normal file
View File

@ -0,0 +1,117 @@
$(function() {
checkLength = function() {
if($tweet.val().length > 140) {
short = $tweet.val().slice(0,140)
$tweet.val(short)
return false
}
}
setLength = function() {
$("#length").val($("#tweetr").val().length+"/140")
}
twoDig = function(d) {
return (d<10) ? "0"+d : d
}
dateStr = function(d) {
//~2014.8.11..21.01.58
return "~" + d.getFullYear() +
"." + twoDig(d.getMonth()+1) +
"." + twoDig(d.getDate()) +
".." + twoDig(d.getHours()) +
"." + twoDig(d.getMinutes()) +
"." + twoDig(d.getSeconds())
}
setTime = function() {
d = new Date()
_datestr = dateStr(d)
$("#twet .date").text(_datestr)
}
setInterval(setTime,1000)
setTime()
$tweet = $("#tweetr")
$time = $("#time")
$submit = $('#submit')
$tweet.focus()
$tweet[0].selectionStart = $tweet[0].selectionEnd = $tweet.val().length
$tweet.change(checkLength)
$tweet.keydown(checkLength)
$tweet.keyup(setLength)
$submit.click(function() {
tweet = $tweet.val()
$tweet.attr('disabled', true)
$submit.attr('disabled', true)
$submit.addClass('disabled')
window.urb.send({
appl:"twit",
data:{tweet:tweet}
}, function(err,res) {
$tweet.attr('disabled', false)
$submit.attr('disabled', false)
$submit.removeClass('disabled')
_tweet = {
created_at: String(new Date()),
text: tweet,
pending: true
}
console.log('set it')
console.log(_tweet)
$time.prepend(renderTweet(_tweet))
$tweet.val('')
setLength()
})
})
renderTweet = function(tweet) {
d = new Date(tweet.created_at)
_datestr = dateStr(d)
css = "tweet"
if(tweet.pending == true)
css += " pending"
$_tweet = $("<div class='"+css+"'></div>")
$_tweet.append("<div class='author'>@urbit_test</div>")
$_tweet.append("<div class='date'>"+_datestr+"</div>")
$_tweet.append("<div class='text'>"+tweet.text+"</div>")
return $_tweet
}
renderTimeline = function(timeline) {
$time.html("")
for(i in timeline) {
$time.append(renderTweet(timeline[i]))
}
}
renderError = function(error) {
$time.html("<div class='error'>Sorry! There was an error fetching from Twitter: "+error+"</div>")
}
window.urb.subscribe({
appl:"twit",
path:"/line"
}, function(err,res) {
console.log('subscr')
console.log(arguments)
console.log((res.data && res.data[0]))
if(err)
return
if(res.data) {
if(res.data[0]) {
renderTimeline(res.data)
}
if(res.errors) {
renderError(res.errors[0])
}
}
})
})

View File

@ -1,415 +1,122 @@
!:
=> |%
++ rev 0
++ axle ,[@ @]
++ gilt :: State data
$% [%json p=json] :: JSON format
[%html p=@t]
[%hymn p=manx] :: HTML format
==
++ gift :: Message data
$% [%rust gilt] :: Refresh webapp
[%rush gilt] :: Update webapp
[%mean (unit (pair term (list tank)))] :: Error, maybe w/ msg
[%nice ~] :: Response message
==
++ move ,[p=bone q=(mold note gift)] :: Arvo command
++ sign
$% $: %e
$% [%thou p=httr]
== == ==
++ note :: Arvo message
$% $: %e
$% [%them p=(unit hiss)]
== == ==
:: A simple Twitter servant.
::
:::: /hook/core/twit/app
::
/? 314 :: need urbit 314
/+ twitter :: use twitter library
/= front /:/%%/front:/hymn/ :: load front page
::
:::: structures
::
|% :: structures
++ axle ,[%0 axle-a] :: application state
++ axle-a ::
%- unit ::
$: las=@da :: update time
txt=@t :: XX timeline text
jon=json :: timeline json
== ::
++ axle-old :: any historic state
$% [%0 axle-a] ::
== ::
++ gilt :: subscription frame
$% [%json p=json] :: json data
[%html p=@t] :: html text
[%hymn p=manx] :: html tree
== ::
++ gift :: output action
$% [%rust gilt] :: total update
[%mean p=ares] :: message failure
[%nice ~] :: succeed
== ::
++ move ,[p=bone q=(mold note gift)] :: output operation
++ sign :: system response
$% $: %e :: from %eyre
$% [%thou p=httr] :: HTTP response
== == == ::
++ note :: system request
$% $: %e :: through %eyre
$% [%them p=(unit hiss)] :: HTTP request
== == == ::
-- ::
::
:::: constants
::
|%
++ hardcoded-key :: hardcoded keys!
:* :- 'hDDOTPfGHGlsOUbhpy6qc6XbW'
'olCkea6wm3XG4pnVCHuPIozUF2ggH1sHjnBtuT4Ai6rCOeQGzO'
:- '2485712317-R77Lpdu5rAJadRVxTXPpnxvcwS0IfNG7QEzLPty'
'a41d83XId0P7QQbodkPYv3zxoEL0Cq9EsN2eXZBZAwAWA'
==
--
|_ [hid=hide vat=axle]
!:
:::: program
::
|_ $: hid=hide :: system state
vat=axle :: custom state
==
++ it :: internals
|%
++ line :: get timeline
(~(stat-home twit hardcoded-key lat.hid `@`eny.hid) ~ ~)
::
++ lint :: publish timeline
^- (list move)
?~ +.vat ~
%+ turn
(skim (~(tap by sup.hid)) |=([* * pax=path] ?=([%line ~] pax)))
|=([ost=bone *] `move`[ost give/rust/json/[jon.u.vat]])
::
++ post :: post a tweet
|= txt=cord
^- hiss
(~(stat-upda twit hardcoded-key lat.hid `@`eny.hid) [%status txt]~ ~)
--
::
++ root :: App root location
/(scot %p our.hid)/main/(scot %da lat.hid)/app/[app.hid]
::
++ incl :: Include scripts
|= wal=wall
%+ turn wal
|= tape ;script(type "text/javascript", src +<);
::
++ peer :: Accept subscribes
++ page front :: build front page
++ prep :: load old state
|= old=(unit (unit axle-old))
[~ +>]
::
++ peer :: accept subscriber
|= [ost=bone you=ship pax=path]
^- [(list move) _+>]
?~ pax
:: [[ost %give %rust %hymn page]~ +>.$]
[[ost %pass /curl1 %e [%them (some timl)]]~ +>.$]
[~ +>.$]
::
++ poke-json
:_ +>.$
?~ pax [ost %give %rust %hymn page]~
?> ?=([%line ~] pax)
?~ +.vat
[ost %pass /line %e [%them (some line:it)]]~
[ost %give %rust %json jon.u.vat]~
::
++ poke-json :: browser message
|= [ost=bone his=ship jon=json]
^- [(list move) _+>]
:: this is where you take the jon and post it to twitter.
~& [%got-json jon]
[[ost %pass /curl2 %e [%them (some (posl jon))]]~ +>.$]
::[[ost %give %nice ~]~ +>.$]
::
++ page
|= a=@t
^- manx
;html
;head
;style:"{(trip ;;(,@ .^(%cx (welp root /main/css))))}"
;* %- incl :~
"//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.1/jquery.min.js"
==
;title: twit
==
;body
;div#c
;div#d
;div#twet
;textarea#tweetr;
;input#submit(type "button", value "Send");
;input#length(type "button", value "0/140");
==
;div#time;
==
==
:: this is where you put the json from the home timeline in
:: make sure its in a script tag with the id jime
;script#jime:"{(trip a)}"
;script:"{(trip ;;(,@ .^(cx/(welp (scag 3 `path`root) /lib/urb/js))))}"
;script:"{(trip ;;(,@ .^(%cx (welp root /main/js))))}"
==
:_ +>.$
=+ txt=+:(need ((of [%tweet so] ~):jo jon))
~& [%tweet txt]
:~ [ost %pass /tweet/(scot %ud ost) %e [%them (some (post:it txt))]]
[ost %give %nice ~]
==
++ pour
|= [way=path sih=sign]
::
++ pour :: HTTP response
|= [pax=path sih=sign]
^- [(list move) _+>]
:_ +>.$
%+ turn (~(tap in (~(get ju pus.hid) ~)))
|= ost=bone
[ost %give %rust %hymn (page `@t`q:(need r.p.sih))]
++ testt
^- hiss
[(scan "http://www.example.com" auri:epur) %get ~ ~]
++ timl
^- hiss
=+ (twit lat.hid `@`eny.hid)
(stat-home ~ ~)
++ posl
|= jon=json
=+ txt=`(unit ,[%tweet p=cord])`((of:jo [%tweet so:jo] ~) jon)
~& [%posl p:(need txt)]
^- hiss
=+ (twit lat.hid `@`eny.hid)
(stat-upda ~[(st ~ p:(need txt))] ~)
++ gsub :: move to zuse?
|= [a=@t b=@t t=@t]
^- @t
?~ t
t
%+ add
(lsh 3 1 $(t (rsh 3 1 t)))
=+ c=(mod t (bex 8))
?:(=(a c) b c)
::
++ req
|= a=tape
~& [%req a]
^- [p=path q=tape r=purl]
[`path`[(scot %t (rap 3 a)) ~] a (scan a auri:epur)]
::
++ app
|% ++ cok 'hDDOTPfGHGlsOUbhpy6qc6XbW'
++ cos 'olCkea6wm3XG4pnVCHuPIozUF2ggH1sHjnBtuT4Ai6rCOeQGzO'
++ aok '2485712317-R77Lpdu5rAJadRVxTXPpnxvcwS0IfNG7QEzLPty'
++ aos 'a41d83XId0P7QQbodkPYv3zxoEL0Cq9EsN2eXZBZAwAWA'
--
::
++ oath ::
|_ [med=meth lur=tape pars=(list ,tape) et=@ en=@]
++ non (turn (rip 2 (shaw et 128 en)) |=(a=@ ~(x ne a)))
++ tim (slag 2 (scow %ui (unt et)))
?+ -.pax !!
%line :: timeline response
?. =(200 p.p.sih)
~& [%timeline-error p.p.sih]
~
=+ txt=`@t`q:(need r.p.sih)
=+ jon=(rash txt apex:poja) :: XX check content!!!
=+ old=&(?=(^ +.vat) =(jon jon.u.vat))
=. +.vat `[lat.hid txt jon]
?:(old ~ lint:it)
::
++ bas ^- tape
=+ ^= hds
%- reel :_
|=([p=tape q=tape] :(weld p "&" q))
%- sort :_ aor
%- weld :- pars
^- (list tape)
:~ :(weld "oauth_consumer_key=" (trip cok:app))
:(weld "oauth_nonce=" non)
:(weld "oauth_signature_method=HMAC-SHA1")
:(weld "oauth_timestamp=" tim)
:(weld "oauth_token=" (trip aok:app))
:(weld "oauth_version=1.0")
==
;: weld
(trip (cuss (trip `@t`med))) "&"
(urle lur) "&"
(urle (scag (dec (lent hds)) `tape`hds))
==
++ sky ^- @t
(crip :(weld (urle (trip cos:app)) "&" (urle (trip aos:app))))
++ sig ^- tape
(sifo (swap 3 (hmac (swap 3 sky) (crip bas))))
++ hed
%- crip
;: weld "OAuth "
"oauth_consumer_key=" "\"" (trip cok:app) "\", "
"oauth_nonce=" "\"" non "\", "
"oauth_signature=" "\"" (urle sig) "\", "
"oauth_signature_method=\"HMAC-SHA1\", "
"oauth_timestamp=" "\"" tim "\", "
"oauth_token=" "\"" (trip aok:app) "\", "
"oauth_version=1.0"
==
--
::
++ help
|= pars=(list ,[tape tape])
^- (list tape)
(turn pars |=(a=[p=tape q=tape] :(weld (urle p.a) "=" (urle q.a))))
++ twyt
|= [med=meth [ur=tape pars=quay] est=time eny=@]
^- hiss
=+ tpar=(turn pars |=([p=@t q=@t] [(trip p) (trip q)]))
=+ hpara=(help (turn tpar |=(p=[p=tape q=tape] [p.p (urle q.p)])))
=+ hparb=(help tpar)
=+ url="https://api.twitter.com/1.1{ur}.json"
=+ token=~(hed oath med url hparb est eny)
=+ ^= head
%- ~(gas by *math)
:~ ['Authorization' [token ~]]
['content-type' ['application/x-www-form-urlencoded' ~]]
==
=+ lvaa=(reel hpara |=([p=tape q=tape] :(weld p "&" q)))
=+ vara=?:(=(0 (lent lvaa)) ~ (scag (dec (lent lvaa)) `tape`lvaa))
=+ lvab=(reel hparb |=([p=tape q=tape] :(weld p "&" q)))
=+ varb=?:(=(0 (lent lvab)) ~ (scag (dec (lent lvab)) `tape`lvab))
?: =(%get med)
?~ vara
[r:(req url) med head ~]
[r:(req :(weld url "?" vara)) med head ~]
=+ car=(crip varb)
=+ ^= bod
^- (unit octs)
(some [(met 3 car) car])
[r:(req url) med head bod]
::
++ twit
|= [est=time eny=@uw]
|%
++ catt
|= [@t @t]
^- @t
(cat 3 +<- +<+)
++ fass
|= a=path
%- trip
%^ gsub '-' '_'
%+ reel a
|= [p=@t q=@t]
:(catt '/' p q)
:: Twitter types
++ tid ,@u
++ lsc (list scr)
++ lst (list ,@t)
++ lid (list tid)
++ gat ,@t :: grant type
++ tok ,@t :: oauth token
++ dev ,@t :: device name, either 'sms' or 'none'
++ nam ,@t :: location name
++ url ,@t :: callback url
++ pla ,@t :: place-id
++ scr ,@t :: screen name
++ slu ,@t :: short list or category name
++ lat ,@t :: latitude
++ lon ,@t :: longitude
:: Parameter types
++ at ,[%access-token p=tok]
++ de ,[%device p=dev]
++ fo ,[%follow p=lid]
++ gr ,[%grant-type p=gat]
++ id ,[%id p=tid]
++ ii ,[%'!inline' p=@t] :: inline
++ is ,[%id p=lid] :: conflict with ++id
++ la ,[%lat p=lat]
++ lo ,[%long p=lon]
++ na ,[%name p=lid]
++ oa ,[%oauth-callback p=url]
++ os ,[%source-screen-name p=scr]
++ pl ,[%place-id p=pla]
++ qq ,[%q p=@t]
++ sc ,[%screen-name p=scr]
++ ss ,[%screen-name p=lsc] :: conflict wiht ++sc
++ sl ,[%slug p=slu]
++ si ,[%source-id p=tid]
++ st ,[%status p=@t]
++ te ,[%text p=@t]
++ ti ,[%target-id p=tid]
++ ts ,[%target-screen-name p=scr]
++ tr ,[%track p=lst]
++ ur ,[%url p=url]
++ ui ,[%user-id p=tid]
++ us ,[%user-id p=lid] :: conflict with ++ui
:: Compound parameter types
++ sid ?(ui sc)
++ lalo ,[p=la q=lo ~]
++ lalona ,[p=la q=lo r=na ~]
++ sidte ,[p=sid q=te ~]
++ sisstiss ,[p=?(si os) q=?(ti ts) ~]
::
++ utt |=(@ `@t`(rsh 3 2 (scot %ui +<)))
++ llsc |=((list scr) `@t`(roll +< |=([p=scr q=@t] (cat 3 (cat 3 q ',') p))))
++ llst |=((list ,@t) `@t`(roll +< |=([p=@t q=@t] (cat 3 (cat 3 q ',') p))))
++ llid |=((list tid) `@t`(roll +< |=([p=tid q=@t] (cat 3 (cat 3 q ',') (utt p)))))
++ clum ,$+(* *)
::
++ funk
|* [med=meth pax=path a=$+(* *)]
|= [args=a opts=quay]
(twyt med (monkey pax args opts) est eny)
++ monkey
|= [pax=path banana=(list ,[p=@t q=?(@ (list ,@))]) opts=quay]
^- [path quay]
?~ banana
[(fass pax) opts]
?: =('!inline' p.i.banana)
?@ q.i.banana
[(fass (welp pax /[`@t`q.i.banana])) opts]
!!
:- (fass pax)
%+ welp opts
%+ turn
(eat banana)
|=([p=@t q=@t] [(gsub '-' '_' p) q])
++ eat
|= food=(list ,[p=@t q=?(@ (list ,@))])
~& [%eat food]
^- quay
%+ turn food
|= [p=@t q=?(@ (list ,@))]
^- [@t @t]
:- `@t`p
^- @t
?@ q
?- p
?(%id %source-id %target-id %user-id) (utt q)
@ `@t`q
==
?- p
?(%follow %id %name %user-id) (llid q)
%track (llst q)
%screen-name (llsc q)
* !!
==
++ stat-ment (funk %get /statuses/mentions-timeline ,~)
++ stat-user (funk %get /statuses/user-timeline ,[sid ~])
++ stat-home (funk %get /statuses/home-timeline ,~)
++ stat-retw (funk %get /statuses/retweets-of-me ,~)
++ stat-rets-iddd (funk %get /statuses/retweets ,[ii ~])
++ stat-show (funk %get /statuses/show ,[id ~])
++ stat-dest-iddd (funk %post /statuses/destroy ,[ii ~])
++ stat-upda (funk %post /statuses/update ,[st ~])
++ stat-retw-iddd (funk %post /statuses/retweet ,[ii ~])
++ stat-oemb-iddd (funk %get /statuses/oembed ,[id ~])
++ stat-oemb-urll (funk %get /statuses/oembed ,[ur ~])
++ stat-retw-idss (funk %get /statuses/retweeters/ids ,[id ~])
++ sear-twee (funk %get /search/tweets ,[qq ~])
++ stat-filt-foll (funk %post /statuses/filter ,[?(fo tr) ~])
++ stat-samp (funk %get /statuses/sample ,~)
++ stat-fire (funk %get /statuses/firehose ,~)
++ user (funk %get /user ,~)
++ site (funk %get /site ,[fo ~])
++ dire (funk %get /direct-messages ,~)
++ dire-sent (funk %get /direct-messages/sent ,~)
++ dire-show (funk %get /direct-messages/show ,[id ~])
++ dire-dest (funk %post /direct-messages/destroy ,[id ~])
++ dire-neww (funk %post /direct-messages/new ,[sid te ~])
++ frie-nore-idss (funk %get /friendships/no-retweets/ids ,~)
++ frie-idss (funk %get /friends/ids ,[sid ~])
++ foll-idss (funk %get /followers/ids ,[sid ~])
++ frie-inco (funk %get /friendships/incoming ,~)
++ frie-outg (funk %get /friendships/outgoing ,~)
++ frie-crea (funk %post /friendships/create ,[sid ~])
++ frie-dest (funk %post /friendships/destroy ,[sid ~])
++ frie-upda (funk %post /friendships/update ,[sid ~])
++ frie-show (funk %get /friendships/show ,[?(si os) ?(ti ts) ~])
++ frie-list (funk %get /friends/list ,[sid ~])
++ foll-list (funk %get /followers/list ,[sid ~])
++ frie-look (funk %get /friendships/lookup ,[?(us ss) ~])
++ acco-sett-gett (funk %get /account/settings ,~)
++ acco-veri (funk %get /account/verify-credentials ,~)
++ acco-sett-post (funk %post /account/settings ,~)
++ acco-upda-deli (funk %post /account/update-delivery-device ,[de ~])
++ acco-upda-prof (funk %post /account/update-profile ,~)
++ acco-upda-prof-back (funk %post /account/update-profile-background-image ,~)
++ acco-upda-prof-colo (funk %post /account/update-profile-colors ,~)
++ bloc-list (funk %get /blocks/list ,~)
++ bloc-idss (funk %get /blocks/ids ,~)
++ bloc-crea (funk %post /blocks/create ,[sid ~])
++ bloc-dest (funk %post /blocks/destroy ,[sid ~])
++ user-look (funk %get /users/lookup ,[?(us ss) ~])
++ user-show (funk %get /users/show ,[sid ~])
++ user-sear (funk %get /users/search ,[qq ~])
++ user-cont-tees (funk %get /users/contributees ,[sid ~])
++ user-cont-tors (funk %get /users/contributors ,[sid ~])
++ acco-remo (funk %post /account/remove-profile-banner ,~)
++ user-prof (funk %get /users/profile-banner ,[sid ~])
++ mute-user-crea (funk %post /mutes/users/create ,[sid ~])
++ mute-user-dest (funk %post /mutes/users/destroy ,[sid ~])
++ mute-user-idss (funk %get /mutes/users/ids ,~)
++ mute-user-list (funk %get /mutes/users/list ,~)
++ user-sugg-slug (funk %get /users/suggestions ,[sl ~])
++ user-sugg (funk %get /users/suggestions ,~)
++ favo-list (funk %get /favorites/list ,~)
++ favo-dest (funk %post /favorites/destroy ,[id ~])
++ favo-crea (funk %post /favorites/create ,[id ~])
++ list-list (funk %get /lists/list ,~)
++ list-stat (funk %get /lists/statuses ,~)
++ list-memb-dest (funk %post /lists/members/destroy ,~)
++ list-memb-hips (funk %get /lists/memberships ,[sid ~])
++ list-subs-bers (funk %get /lists/subscribers ,~)
++ list-subs-crea (funk %post /lists/subscribers/create ,~)
++ list-subs-show (funk %get /lists/subscribers/show ,[sid ~])
++ list-subs-dest (funk %post /lists/subscribers/destroy ,~)
++ list-memb-crea-alll (funk %post /lists/members/create-all ,[?(us ss) ~])
++ list-memb-show (funk %get /lists/members/show ,[sid ~])
++ list-memb-bers (funk %get /lists/members ,~)
++ list-memb-crea (funk %post /lists/members/create ,[sid ~])
++ list-dest (funk %post /lists/destroy ,~)
++ list-upda (funk %post /lists/update ,~)
++ list-crea (funk %post /lists/create ,[na ~])
++ list-show (funk %get /lists/show ,~)
++ list-subs-ions (funk %get /lists/subscriptions ,[sid ~])
++ list-memb-dest-alll (funk %post /lists/members/destroy-all ,[?(us ss) ~])
++ list-owne (funk %get /lists/ownerships ,[sid ~])
++ save-list (funk %get /saved-searches/list ,~)
++ save-show-iddd (funk %get /saved-searches/show ,[ii ~])
++ save-crea (funk %post /saved-searches/create ,[qq ~])
++ save-dest-iddd (funk %post /saved-searches/destroy ,[ii ~])
++ geoo-iddd-plac (funk %get /geo/id ,[ii ~])
++ geoo-reve (funk %get /geo/reverse-geocode ,[la lo ~])
++ geoo-sear (funk %get /geo/search ,~)
++ geoo-simi (funk %get /geo/similar-places ,[la lo na ~])
++ tren-plac (funk %get /trends/place ,[id ~])
++ tren-avai (funk %get /trends/available ,~)
++ tren-clos (funk %get /trends/closest ,[la lo ~])
++ user-repo (funk %post /users/report-spam ,[sid ~])
++ oaut-auth-cate (funk %get /oauth/authenticate ,~)
++ oaut-auth-rize (funk %get /oauth/authorize ,~)
++ oaut-acce (funk %post /oauth/access-token ,~)
++ oaut-requ (funk %post /oauth/request-token ,[oa ~])
++ oaut-toke (funk %post /oauth2/token ,[gr ~])
++ oaut-inva (funk %post /oauth2/invalidate-token ,[at ~])
++ help-conf (funk %get /help/configuration ,~)
++ help-lang (funk %get /help/languages ,~)
++ help-priv (funk %get /help/privacy ,~)
++ help-toss (funk %get /help/tos ,~)
++ appl-rate (funk %get /application/rate-limit-status ,~)
++ stat-look (funk %get /statuses/lookup ,[us ~])
%tweet :: post response
=+ ost=(slav %ud -.+.pax)
~? !=(200 p.p.sih) [%tweet-error p.p.sih]
[ost %pass /line %e [%them (some line:it)]]~
==
--
--

View File

@ -0,0 +1,39 @@
:: Front page of the twitter app.
::
:::: /hook/hymn/front/twit/app
::
/? 314 :: need urbit 314
/= urbit /:/===/lib/urb:/hymn/ :: urbit library (js)
/= style /:/%%%/style:/hymn/ :: stylesheet (css)
/= application /:/%%%/app:/hymn/ :: application (js)
!:
:::: content
::
^- manx
;html
;head
;title: Urbit - Twitter Test
;+ style
;script
=type "text/javascript"
=src "//cdnjs.cloudflare.com/ajax/libs/jquery/2.1.1/jquery.min.js"
;
==
==
;body
;div#c
;div#d
;div#twet
;div.author: @urbit_test
;div.date;
;textarea#tweetr(placeholder "What would you like to tweet?");
;input#submit(type "button", value "Send");
;input#length(type "button", value "0/140");
==
;div#time: Fetching...
==
==
;+ urbit
;+ application
==
==

126
main/app/twit/style.css Normal file
View File

@ -0,0 +1,126 @@
body,
textarea,
input {
font-family: "Helvetica Neue", Helvetica, Arial, sans-serif;
font-weight: 400;
}
body {
margin-top: 4rem;
font-size: 18px;
}
input,
textarea {
font-size: 1rem;
outline: none;
}
#c {
width: 100%;
height: 100%;
}
#d,
#tweet,
.tweet,
.text,
textarea {
width: 32rem;
}
#d {
margin-left: auto;
margin-right: auto;
}
textarea {
border: 0;
height: 8rem;
line-height: 1.5rem;
margin-bottom: .3rem;
resize: none;
padding: 1rem 1px;
background-color: #f7f7f7;
}
textarea:focus {
background-color: #eee;
}
#length,
#submit {
border: 0;
color: #333;
letter-spacing: 0.01rem;
}
#length {
width 1rem;
height 1rem;
font-size: .6rem;
color: #ccc;
background-color: transparent;
}
#submit {
background-color: transparent;
border: 2px solid #5DE668;
color: #5DE668;
padding: .3rem 1rem;
font-weight: 500;
cursor: pointer;
}
#submit:hover,
#submit:focus {
background-color: #5DE668;
color: #fff;
}
#submit.disabled {
opacity: .6;
}
.text {
word-wrap:break-word;
margin-bottom: .3rem;
line-height: 1.6rem;
}
#twet {
margin-bottom: 3rem;
border-bottom: 2px solid #464646;
padding-bottom: 2rem;
}
#twet,
.tweet {
border-bottom: 1px solid #eee;
}
.tweet {
margin-bottom: 2rem;
padding-bottom: 1rem;
}
.author,
.date {
font-size: .8rem;
color: #ccc;
margin-right: .6rem;
display: inline-block;
margin-bottom: .3rem;
}
.author {
color: #888;
}
.pending {
opacity: .3;
}
.error {
color: #FF5F5F;
letter-spacing: .06rem;
}

525
main/lib/twitter/core.hook Normal file
View File

@ -0,0 +1,525 @@
:: A Twitter API library.
::
:::: /hook/core/twitter/lib
::
/? 314
::
:::: structures
::
|%
++ keys :: twitter-key type
$: con=[tok=@t sec=@t] :: user key pair
acc=[tok=@t sec=@t] :: app key pair
==
--
!:
:::: functions
::
|%
++ fass :: rewrite path
|= a=path
%- trip
%^ gsub '-' '_'
%+ reel a
|= [p=@t q=@t]
(cat 3 '/' (cat 3 p q))
::
++ gsub :: replace chars
|= [a=@t b=@t t=@t]
^- @t
?~ t t
%+ add (lsh 3 1 $(t (rsh 3 1 t)))
=+ c=(mod t (bex 8))
?:(=(a c) b c)
::
++ oauth :: OAuth 1.0 header
|= $: med=meth
url=tape
pas=(list tape)
key=keys
zet=@
ken=@
==
^- @t
=+ non=(turn (rip 2 (shaw zet 128 ken)) |=(a=@ ~(x ne a)))
=+ tim=(slag 2 (scow %ui (unt zet)))
=+ sky=(crip :(weld (urle (trip sec.con.key)) "&" (urle (trip sec.acc.key))))
=+ ^= bas
^- tape
=+ ^= hds
%- reel :_ |=([p=tape q=tape] :(weld p "&" q))
%- sort :_ aor
%- weld :- pas
^- (list tape)
:~ :(weld "oauth_consumer_key=" (trip tok.con.key))
:(weld "oauth_nonce=" non)
:(weld "oauth_signature_method=HMAC-SHA1")
:(weld "oauth_timestamp=" tim)
:(weld "oauth_token=" (trip tok.acc.key))
:(weld "oauth_version=1.0")
==
;: weld
(trip (cuss (trip `@t`med))) "&"
(urle url) "&"
(urle (scag (dec (lent hds)) `tape`hds))
==
=+ sig=`tape`(sifo (swap 3 (hmac (swap 3 sky) (crip bas))))
%- crip
;: weld "OAuth "
"oauth_consumer_key=" "\"" (trip tok.con.key) "\", "
"oauth_nonce=" "\"" non "\", "
"oauth_signature=" "\"" (urle sig) "\", "
"oauth_signature_method=\"HMAC-SHA1\", "
"oauth_timestamp=" "\"" tim "\", "
"oauth_token=" "\"" (trip tok.acc.key) "\", "
"oauth_version=1.0"
==
::
++ valve :: produce request
|= $: med=meth
[rus=tape quy=quay]
key=keys
est=time
eny=@
==
^- hiss
=+ url="https://api.twitter.com/1.1{rus}.json"
=+ req=|=(a=tape (scan a auri:epur))
=+ ^= help
|= quy=(list ,[tape tape])
^- (list tape)
%+ turn quy
|= a=[p=tape q=tape]
:(weld (urle p.a) "=" (urle q.a))
=+ tan=(turn quy |=([p=@t q=@t] [(trip p) (trip q)]))
=+ har=(help (turn tan |=(p=[p=tape q=tape] [p.p (urle q.p)])))
=+ hab=(help tan)
=+ lav=(reel har |=([p=tape q=tape] :(weld p "&" q)))
=+ voy=?:(=(0 (lent lav)) ~ (scag (dec (lent lav)) `tape`lav))
=+ vab=(reel hab |=([p=tape q=tape] :(weld p "&" q)))
=+ vur=(crip ?:(=(0 (lent vab)) ~ (scag (dec (lent vab)) `tape`vab)))
=+ ^= head
%- ~(gas by *math)
:~ ['authorization' [(oauth med url hab key est eny) ~]]
['content-type' ['application/x-www-form-urlencoded' ~]]
==
?: =(%get med)
?~ voy
[(req url) med head ~]
[(req :(weld url "?" voy)) med head ~]
[(req url) med head (some [(met 3 vur) vur])]
--
!:
:::: library
::
|%
++ twit
=> |% :: request structures
++ dev ,@t :: device name
++ gat ,@t :: grant type
++ lat ,@t :: latitude
++ lid (list tid)
++ lon ,@t :: longitude
++ lsc (list scr)
++ lst (list ,@t)
++ nam ,@t :: location name
++ pla ,@t :: place-id
++ scr ,@t :: screen name
++ slu ,@t :: category name
++ tid ,@u
++ tok ,@t :: oauth token
++ url ,@t :: callback url
::
++ at ,[%access-token p=tok]
++ de ,[%device p=dev]
++ fo ,[%follow p=lid]
++ gr ,[%grant-type p=gat]
++ id ,[%id p=tid]
++ ii ,[%'!inline' p=@t]
++ is ,[%id p=lid]
++ la ,[%lat p=lat]
++ lo ,[%long p=lon]
++ na ,[%name p=lid]
++ oa ,[%oauth-callback p=url]
++ os ,[%source-screen-name p=scr]
++ pl ,[%place-id p=pla]
++ qq ,[%q p=@t]
++ sc ,[%screen-name p=scr]
++ sd ?(ui sc)
++ ss ,[%screen-name p=lsc]
++ sl ,[%slug p=slu]
++ si ,[%source-id p=tid]
++ st ,[%status p=@t]
++ te ,[%text p=@t]
++ ti ,[%target-id p=tid]
++ ts ,[%target-screen-name p=scr]
++ tr ,[%track p=lst]
++ ur ,[%url p=url]
++ ui ,[%user-id p=tid]
++ us ,[%user-id p=lid]
--
|_ [key=keys est=time eny=@uw]
++ lutt |=(@ `@t`(rsh 3 2 (scot %ui +<)))
++ llsc
|= (list scr)
(roll +< |=([p=scr q=@t] (cat 3 (cat 3 q ',') p)))
::
++ llst
|= (list ,@t)
(roll +< |=([p=@t q=@t] (cat 3 (cat 3 q ',') p)))
::
++ llid
|= (list tid)
(roll +< |=([p=tid q=@t] (cat 3 (cat 3 q ',') (lutt p))))
::
++ mold :: construct request
|* [med=meth pax=path a=$+(* *)]
|= [args=a quy=quay]
(valve med (cowl pax args quy) key est eny)
::
++ cowl :: handle parameters
|= $: pax=path
ban=(list ,[p=@t q=?(@ (list ,@))])
quy=quay
==
^- [path quay]
?~ ban
[(fass pax) quy]
?: =('!inline' p.i.ban)
?@ q.i.ban
[(fass (welp pax /[`@t`q.i.ban])) quy]
!!
:- (fass pax)
%+ welp quy
%+ turn `(list ,[p=@t q=?(@ (list ,@))])`ban
|= [p=@t q=?(@ (list ,@))]
^- [@t @t]
:- (gsub '-' '_' p)
?@ q
?- p
?(%id %source-id %target-id %user-id) (lutt q)
@ `@t`q
==
?- p
?(%follow %id %name %user-id) (llid q)
%track (llst q)
%screen-name (llsc q)
* !!
==
::
++ stat-ment
(mold %get /statuses/mentions-timeline ,~)
::
++ stat-user
(mold %get /statuses/user-timeline ,[sd ~])
::
++ stat-home
(mold %get /statuses/home-timeline ,~)
::
++ stat-retw
(mold %get /statuses/retweets-of-me ,~)
::
++ stat-rets-iddd
(mold %get /statuses/retweets ,[ii ~])
::
++ stat-show
(mold %get /statuses/show ,[id ~])
::
++ stat-dest-iddd
(mold %post /statuses/destroy ,[ii ~])
::
++ stat-upda
(mold %post /statuses/update ,[st ~])
::
++ stat-retw-iddd
(mold %post /statuses/retweet ,[ii ~])
::
++ stat-oemb-iddd
(mold %get /statuses/oembed ,[id ~])
::
++ stat-oemb-urll
(mold %get /statuses/oembed ,[ur ~])
::
++ stat-retw-idss
(mold %get /statuses/retweeters/ids ,[id ~])
::
++ sear-twee
(mold %get /search/tweets ,[qq ~])
::
++ stat-filt-foll
(mold %post /statuses/filter ,[?(fo tr) ~])
::
++ stat-samp
(mold %get /statuses/sample ,~)
::
++ stat-fire
(mold %get /statuses/firehose ,~)
::
++ user
(mold %get /user ,~)
::
++ site
(mold %get /site ,[fo ~])
::
++ dire
(mold %get /direct-messages ,~)
::
++ dire-sent
(mold %get /direct-messages/sent ,~)
::
++ dire-show
(mold %get /direct-messages/show ,[id ~])
::
++ dire-dest
(mold %post /direct-messages/destroy ,[id ~])
::
++ dire-neww
(mold %post /direct-messages/new ,[sd te ~])
::
++ frie-nore-idss
(mold %get /friendships/no-retweets/ids ,~)
::
++ frie-idss
(mold %get /friends/ids ,[sd ~])
::
++ foll-idss
(mold %get /followers/ids ,[sd ~])
::
++ frie-inco
(mold %get /friendships/incoming ,~)
::
++ frie-outg
(mold %get /friendships/outgoing ,~)
::
++ frie-crea
(mold %post /friendships/create ,[sd ~])
::
++ frie-dest
(mold %post /friendships/destroy ,[sd ~])
::
++ frie-upda
(mold %post /friendships/update ,[sd ~])
::
++ frie-show
(mold %get /friendships/show ,[?(si os) ?(ti ts) ~])
::
++ frie-list
(mold %get /friends/list ,[sd ~])
::
++ foll-list
(mold %get /followers/list ,[sd ~])
::
++ frie-look
(mold %get /friendships/lookup ,[?(us ss) ~])
::
++ acco-sett-gett
(mold %get /account/settings ,~)
::
++ acco-veri
(mold %get /account/verify-credentials ,~)
::
++ acco-sett-post
(mold %post /account/settings ,~)
::
++ acco-upda-deli
(mold %post /account/update-delivery-device ,[de ~])
::
++ acco-upda-prof
(mold %post /account/update-profile ,~)
::
++ acco-upda-prof-back
(mold %post /account/update-profile-background-image ,~)
::
++ acco-upda-prof-colo
(mold %post /account/update-profile-colors ,~)
::
++ bloc-list
(mold %get /blocks/list ,~)
::
++ bloc-idss
(mold %get /blocks/ids ,~)
::
++ bloc-crea
(mold %post /blocks/create ,[sd ~])
::
++ bloc-dest
(mold %post /blocks/destroy ,[sd ~])
::
++ user-look
(mold %get /users/lookup ,[?(us ss) ~])
::
++ user-show
(mold %get /users/show ,[sd ~])
::
++ user-sear
(mold %get /users/search ,[qq ~])
::
++ user-cont-tees
(mold %get /users/contributees ,[sd ~])
::
++ user-cont-tors
(mold %get /users/contributors ,[sd ~])
::
++ acco-remo
(mold %post /account/remove-profile-banner ,~)
::
++ user-prof
(mold %get /users/profile-banner ,[sd ~])
::
++ mute-user-crea
(mold %post /mutes/users/create ,[sd ~])
::
++ mute-user-dest
(mold %post /mutes/users/destroy ,[sd ~])
::
++ mute-user-idss
(mold %get /mutes/users/ids ,~)
::
++ mute-user-list
(mold %get /mutes/users/list ,~)
::
++ user-sugg-slug
(mold %get /users/suggestions ,[sl ~])
::
++ user-sugg
(mold %get /users/suggestions ,~)
::
++ favo-list
(mold %get /favorites/list ,~)
::
++ favo-dest
(mold %post /favorites/destroy ,[id ~])
::
++ favo-crea
(mold %post /favorites/create ,[id ~])
::
++ list-list
(mold %get /lists/list ,~)
::
++ list-stat
(mold %get /lists/statuses ,~)
::
++ list-memb-dest
(mold %post /lists/members/destroy ,~)
::
++ list-memb-hips
(mold %get /lists/memberships ,[sd ~])
::
++ list-subs-bers
(mold %get /lists/subscribers ,~)
::
++ list-subs-crea
(mold %post /lists/subscribers/create ,~)
::
++ list-subs-show
(mold %get /lists/subscribers/show ,[sd ~])
::
++ list-subs-dest
(mold %post /lists/subscribers/destroy ,~)
::
++ list-memb-crea-alll
(mold %post /lists/members/create-all ,[?(us ss) ~])
::
++ list-memb-show
(mold %get /lists/members/show ,[sd ~])
::
++ list-memb-bers
(mold %get /lists/members ,~)
::
++ list-memb-crea
(mold %post /lists/members/create ,[sd ~])
::
++ list-dest
(mold %post /lists/destroy ,~)
::
++ list-upda
(mold %post /lists/update ,~)
::
++ list-crea
(mold %post /lists/create ,[na ~])
::
++ list-show
(mold %get /lists/show ,~)
::
++ list-subs-ions
(mold %get /lists/subscriptions ,[sd ~])
::
++ list-memb-dest-alll
(mold %post /lists/members/destroy-all ,[?(us ss) ~])
::
++ list-owne
(mold %get /lists/ownerships ,[sd ~])
::
++ save-list
(mold %get /saved-searches/list ,~)
::
++ save-show-iddd
(mold %get /saved-searches/show ,[ii ~])
::
++ save-crea
(mold %post /saved-searches/create ,[qq ~])
::
++ save-dest-iddd
(mold %post /saved-searches/destroy ,[ii ~])
::
++ geoo-iddd-plac
(mold %get /geo/id ,[ii ~])
::
++ geoo-reve
(mold %get /geo/reverse-geocode ,[la lo ~])
::
++ geoo-sear
(mold %get /geo/search ,~)
::
++ geoo-simi
(mold %get /geo/similar-places ,[la lo na ~])
::
++ tren-plac
(mold %get /trends/place ,[id ~])
::
++ tren-avai
(mold %get /trends/available ,~)
::
++ tren-clos
(mold %get /trends/closest ,[la lo ~])
::
++ user-repo
(mold %post /users/report-spam ,[sd ~])
::
++ oaut-auth-cate
(mold %get /oauth/authenticate ,~)
::
++ oaut-auth-rize
(mold %get /oauth/authorize ,~)
::
++ oaut-acce
(mold %post /oauth/access-token ,~)
::
++ oaut-requ
(mold %post /oauth/request-token ,[oa ~])
::
++ oaut-toke
(mold %post /oauth2/token ,[gr ~])
::
++ oaut-inva
(mold %post /oauth2/invalidate-token ,[at ~])
::
++ help-conf
(mold %get /help/configuration ,~)
::
++ help-lang
(mold %get /help/languages ,~)
::
++ help-priv
(mold %get /help/privacy ,~)
::
++ help-toss
(mold %get /help/tos ,~)
::
++ appl-rate
(mold %get /application/rate-limit-status ,~)
::
++ stat-look
(mold %get /statuses/lookup ,[us ~])
--
--

View File

@ -1,3 +1,8 @@
|_ mud=@t
++ grow |% ++ mime [/text/css (taco mud)] --
++ grow
|% ++ mime [/text/css (taco mud)]
++ hymn ;style
;- (trip mud)
==
--
--

12
main/mar/hook/door.hook Normal file
View File

@ -0,0 +1,12 @@
::
:::: /hook/door/hook/mar
::
/? 314
|_ own=@t
::
++ grow :: convert to
|%
++ mime [/text/html (taco own)] :: convert to %mime
++ hymn ;div:(pre:"{(trip own)}") :: convert to %html
--
--

View File

@ -1,3 +1,8 @@
|_ mud=@t
++ grow |% ++ mime [/application/javascript (taco mud)] --
++ grow
|% ++ mime [/application/javascript (taco mud)]
++ hymn ;script
;- (trip mud)
==
--
--

View File

@ -1,75 +0,0 @@
::
::
::::
::
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div(class "page documentation")
;h1(class "page-title"): Documentation
;p ; If you want to build a deep understanding of how Urbit works,
; start with Nock. If you would prefer to just try stuff
; out, start with Arvo.
==
;p ; This documentation is a work in progress. Feedback and corrections
; are welcome. Pull requests are encouraged. The repo for this site
; lives ;{a(href "https://github.com/urbit/urbit.github.io") "here"}.
; We would love your help in making this reference more useful.
==
;p ; Arvo is still actively being changed and updated.
; As Arvo development cools, more documentation will emerge.
==
;ul
;li
;p:(h2:"Tutorial")
;strong: Nock
;ol
;li:(p:(a/"/gen/main/pub/fab/site/tut/nock1":"Intro to Nock"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/nock2":"Nock is Easy"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/nock3":"Using Nock"))
==
;strong: Hoon
;ol
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon1":"Intro to Hoon"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon2":"Types"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon3":"Hoon Computes"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon4":"Gates"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon5":"Tiles"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon6":"Type Inference"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon7":"Odds, Ends, Quirks"))
==
;strong: Arvo
;ol
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo1":"Basic Arvo"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo2":"More Basic Arvo"))
==
==
;li
;p:(h2:"Reference")
;ul
;li:(p:(a/"/gen/main/pub/fab/site/ref/foreword":";{strong "Foreword"}: Nock"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/preface":";{strong "Preface"}: Hoon Abstract"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/vol0":";{strong "Volume 0"}: Version Stub"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/vol1":";{strong "Volume 1"}: Hoon Structures"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/vol2":";{strong "Volume 2"}: Hoon Compiler"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/vol3":";{strong "Volume 3"}: Arvo Core"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/vol4":";{strong "Volume 4"}: Arvo Vanes"))
;li:(p:(a/"/gen/main/pub/fab/site/ref/postface":";{strong "Postface"}"))
==
==
==
==
==
==
==

View File

@ -1,26 +1,67 @@
:: Top level of the Urbit documentation.
::
:::: /hook/site/src/pub
::
::::
::
/= bod /^ manx /: /===/pub/src/doc/intro /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/intro /psal/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;div(class "page documentation")
;+ bod
;h1(class "page-title"): Documentation
;p ; This documentation is a work in progress. Feedback and corrections
; are welcome. Pull requests are encouraged. The repo for this site
; lives ;{a(href "https://github.com/urbit/urbit.github.io") "here"}.
==
;ul
;li
;p:(h2:"Tutorial")
;p:(a/"/gen/main/pub/fab/site/tut/setup":"Setup")
;strong: Nock
;ol
;li:(p:(a/"/gen/main/pub/fab/site/tut/nock1":"Intro to Nock"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/nock2":"Nock is Easy"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/nock3":"Using Nock"))
==
;strong: Hoon
;ol
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon1":"Intro to Hoon"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon2":"Types"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon3":"Hoon Computes"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon4":"Gates"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon5":"Tiles"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon6":"Type Inference"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/hoon7":"Odds, Ends, Quirks"))
==
;strong: Arvo
;ol
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo1":"Basic Arvo"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo2":"More Basic Arvo"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo3":"Apps"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo4":"Functional Publishing I"))
;li:(p:(a/"/gen/main/pub/fab/site/tut/arvo5":"Functional Publishing II"))
==
==
;li
;p:(h2:"Reference")
;p:(a/"/gen/main/pub/fab/site/ref/foreword":";{strong "Foreword"}: Nock")
;p:(a/"/gen/main/pub/fab/site/ref/preface":";{strong "Preface"}: Hoon Abstract")
;p:(a/"/gen/main/pub/fab/site/ref/vol0":";{strong "Volume 0"}: Version Stub")
;p:(a/"/gen/main/pub/fab/site/ref/vol1":";{strong "Volume 1"}: Hoon Structures")
;p:(a/"/gen/main/pub/fab/site/ref/vol2":";{strong "Volume 2"}: Hoon Compiler")
;p:(a/"/gen/main/pub/fab/site/ref/vol3":";{strong "Volume 3"}: Arvo Core")
;p:(a/"/gen/main/pub/fab/site/ref/vol4":";{strong "Volume 4"}: Arvo Vanes")
==
==
==
==
;+ fot
==
==
==

View File

@ -1,8 +1,6 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/ref/nock /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -12,12 +10,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= bod /^ manx /: /===/pub/src/doc/ref/vol0 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/ref/vol1 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/ref/vol2 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= bod /^ manx /: /===/pub/src/doc/ref/vol0 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/ref/vol3 /psal/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= bod /^ manx /: /===/pub/src/doc/ref/vol0 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/ref/vol4 /psal/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -48,6 +48,7 @@ a:focus {
li a:hover, p a:hover,
li a:hover, p a:focus {
text-decoration: none;
border-bottom: 2px solid #555;
}

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/arvo/1 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/arvo/2 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -0,0 +1,18 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/arvo/app /psal/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;div(class "content container")
;div.subpage
;+ bod
==
==
==
==

View File

@ -0,0 +1,18 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/arvo/pub2 /psal/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;div(class "content container")
;div.subpage
;+ bod
==
==
==
==

View File

@ -0,0 +1,18 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/arvo/pub3 /psal/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;div(class "content container")
;div.subpage
;+ bod
==
==
==
==

View File

@ -0,0 +1,18 @@
/= bod /^ manx /: /===/app/twit/core /hymn/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;div(class "content container")
;div.subpage
;+ bod
==
==
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/1 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/2 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/3 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/4 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/5 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/6 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/hoon/7 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= bod /^ manx /: /===/pub/src/doc/say/nock/1 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= bod /^ manx /: /===/pub/src/doc/say/nock/2 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -1,7 +1,5 @@
/= sty /^ @t /: /===/pub/fab/site/styles /css/
/= bod /^ manx /: /===/pub/src/doc/say/nock/3 /psal/
/= sid /^ manx /: /===/pub/src/site/res/sidebar /hymn/
/= fot /^ manx /: /===/pub/src/site/res/footer /hymn/
/= sty /^ @t /: /===/pub/src/site/res/styles /css/
::
:::: ~tomsyt-balsen
::
@ -11,12 +9,10 @@
;style:"{(trip sty)}"
==
;body
;+ sid
;div(class "content container")
;div.subpage
;+ bod
==
==
;+ fot
==
==

View File

@ -0,0 +1,18 @@
/= bod /^ manx /: /===/pub/src/doc/say/setup /psal/
/= sty /^ @t /: /===/pub/fab/site/styles /css/
::
:::: ~tomsyt-balsen
::
;html
;head
;title: Urbit: Personal Cloud Computing
;style:"{(trip sty)}"
==
;body
;div(class "content container")
;div.subpage
;+ bod
==
==
==
==

View File

@ -1,55 +0,0 @@
#What is Urbit?
We got tired of system software from the 1970s. So we wrote our own. From scratch.
##Nock, a minimal virtual machine
[Nock](https://github.com/urbit/urbit/blob/master/urb/zod/spec/nock/5.txt) is a homoiconic combinator algebra, not much fancier than SKI combinators. The spec fits on a T-shirt and gzips to 340 bytes. Nock's data model is simple: a noun is an atom or a cell. An atom is any natural number. A cell is an ordered pair of any two nouns.
Nock cannot create cycles and needs no GC. Its only arithmetic operator is increment, so it is not inherently efficient; an efficient Nock interpreter works by recognizing standard Nock formulas and matching them to pre-written manual optimizations. We never extend Nock or call out to Unix from it.
##Hoon, a typed functional language
Hoon is a strict, typed, functional language that compiles itself to Nock. The Hoon compiler is 4000 lines of Hoon. Adding standard libraries, the self-compiling kernel is 8000 lines.
Hoon has no particular familial relationship to other languages you may know. It uses its own type inference algorithm and is as different from Haskell as from Lisp. Hoon syntax is also completely unfamiliar. Hoon has the same relationship to Nock that C has to assembly; as thin a layer as possible. It is possible to learn Hoon without Nock, but it's probably not a good idea.
As a functional systems language, Hoon is especially good at metaprogramming, self-virtualization, hotpatching; marshalling and validating untyped data; decoding and encoding binary message formats. Hoon is designed for event programming, so there is no concurrency model.
##Arvo, a functional operating system
Arvo is an event-driven server OS built on the same event library as node.js (libuv). Unlike node.js, Arvo is written in Hoon, isolated from Unix and a persistent single-level store. Arvo can update itself and its data and programs indefinitely from its own state. After the image is created, it lives indefinitely and is never modified externally.
Arvo is modular. Present modules provide a network messaging protocol, a REPL and task manager, a revision-controlled filesystem, a text console, and an HTTP server. The Arvo codebase is about 6500 lines, which is probably too big.
##Clay, a global immutable namespace
Clay is a feature of Arvo: a secure, referentially transparent and decentralized filesystem. The root of the namespace is a global PKI. Each identity serves its own filesystem tree. Files are nouns, not atoms, with arbitrary tree structure.
Identities, which are also routable addresses, are either key fingerprints or hierarchically assigned short numbers. All numbers are mapped to phonemic strings for memorability. Identities are transferrable and irrevocable; ownership is established by key pinning. The system has a superficial resemblance to Bitcoin, but works more like land than money. It does not use mining or a blockchain.
Since Clay is immutable, all paths contain a revision, which is either a change number, a date, or a label. A request is a special case of a subscription, so syndication is straightforward. Clay also has a synthesis mode which computes a functional namespace on the client side.
##What is Urbit good for?
Urbit is good for everything, of course! It is a new general-purpose computing layer. We look forward to a future where the current Internet exists only as an underground series of tubes which transports you to your Urbit.
More specifically, Urbit is a personal cloud computer. Right now, the cloud computers we use run OSes designed for minicomputers in the '70s. An ordinary user can no more drive a Linux box in the cloud than fly an A320. So she has to sit in coach class as a row in someone else's database. It's definitely air travel. It's not exactly flying.
The user of the future will fly her own computer. She will own and control her own identity and her own data. She will even host her own apps. She will not be part of someone else's Big Data. She will be her own Little Data. Unless she's a really severe geek, she will pay some service to store and execute her Urbit ship - but she can move it anywhere else, anytime, for the cost of the bandwidth.
A user can't manage a general-purpose computer unless she basically understands what it's doing. She may not be a programmer, but she needs at least a rough mental model of her computer's state.
A personal computer has to be a *simple* computer. This is why we built a new system software stack from scratch, with the goal of bringing it in under 10,000 lines of code. Urbit is about 50% over this complexity budget, but nobody's perfect.
##What can it do now?
Urbit at present is not good for anything but screwing around. For screwing around, it answers all your needs and is happy to consume any amount of unwanted time.
It does basically work as described above, though. Urbit at present propagates all its updates through its own filesystem, runs its own chat server over its own network protocol, etc. As of early 2014, it is an interesting prototype, not a useful device.
##How can I join the revolution?
Just wait. If you're cool enough, someone will send you an invitation.
Psych! No, actually, you can go here.

File diff suppressed because it is too large Load Diff

View File

@ -1,90 +1,90 @@
volume 3, Arvo models and skeleton
#volume 3, Arvo models and skeleton
++ arch
++ bead
++ care
++ case
++ desk
++ cage
++ chop
++ curd
++ disk
++ duct
++ gene
++ glob
++ herd
++ hilt
++ hypo
++ khan
++ lens
++ v
++ w
++ x
++ y
++ z
++ logo
++ lode
++ mark
++ mill
++ milt
++ monk
++ mold
++ muse
++ mosh
++ move
++ ovum
++ pane
++ pone
++ ship
++ sled
++ slut
++ vile
++ wire
++ slod
++ slub
++ arch
++ bead
++ care
++ case
++ desk
++ cage
++ chop
++ curd
++ disk
++ duct
++ gene
++ glob
++ herd
++ hilt
++ hypo
++ khan
++ lens
++ v
++ w
++ x
++ y
++ z
++ logo
++ lode
++ mark
++ mill
++ milt
++ monk
++ mold
++ muse
++ mosh
++ move
++ ovum
++ pane
++ pone
++ ship
++ sled
++ slut
++ vile
++ wire
++ slod
++ slub
section 3bE, Arvo core
##section 3bE, Arvo core
++ vent
++ ruck
++ wink
++ doze
++ sike
++ souk
++ sunk
++ song
++ spuc
++ sump
++ said
++ scry
++ soar
++ swim
++ vint
++ viol
++ is
++ beck
++ dink
++ dint
++ doos
++ hurl
++ race
++ fire
++ jack
++ kick
++ vent
++ ruck
++ wink
++ doze
++ sike
++ souk
++ sunk
++ song
++ spuc
++ sump
++ said
++ scry
++ soar
++ swim
++ vint
++ viol
++ is
++ beck
++ dink
++ dint
++ doos
++ hurl
++ race
++ fire
++ jack
++ kick
Postface
##Postface
++ come
++ keep
++ load
++ peek
++ poke
++ wish
++ come
++ keep
++ load
++ peek
++ poke
++ vega
++ veer
++ wish
++ come
++ keep
++ load
++ peek
++ poke
++ wish
++ come
++ keep
++ load
++ peek
++ poke
++ vega
++ veer
++ wish

View File

@ -0,0 +1,987 @@
Ames
====
Ames is our networking protocol.
data models
-----------
###`++fort`, formal state
```
++ fort :: formal state
$: %0 :: version
gad=duct :: client interface
hop=@da :: network boot date
ton=town :: security
zac=(map ship corn) :: flows by server
== ::
```
This is the state of our vane. Anything that must be remembered between
calls to ames must be stored in this state.
`%0` is the version of the ames state model itself. If the data model `++fort`
changes, then this number needs to be incremented, and an adapter must be
written to upgrade the old state into the new state. Note that this is the
version number of the model itself, not the contents. When the data changes,
there is of course no need to change this.
`gad` is a `duct` over which we send `%send` cards to unix. This card is
initialized when unix sends a `%barn` card as vere starts up. Vere treats this
duct specially -- don't send anything weird over it.
`hop` is the network boot date. This is set when the `%kick` card is sent by
vere on start up.
`ton` is a `++town`, where we store all of our security/encryption state. Note
that this is shared across all ships on a pier.
`zac` is a map of ships to `++corn`. This stores all the per-ship state. The
keys to this map are the ships on the current pier.
###`++town`, all security state
```
++ town :: all security state
$: lit=@ud :: imperial modulus
any=@ :: entropy
urb=(map ship sufi) :: all keys and routes
fak=? ::
== ::
```
This is the security state of our pier.
`lit` is unused.
`any` is 256 bits of entropy. This entropy is used and updated in exactly two
places: when we send a `%junk` card, and when we generate a new symmetric key
in `++griz:lax:as:go`. When it is updated, it is updated by a SHA-256 hash of
the current time and the old value of the entropy.
`urb` is a map of ships to `++sufi`. This is where we store all the per-ship
state for the pier. The keys to this map are the ships on the current pier.
`fak` is true if we are on a fake network. This disables certain security
checks so that anyone may run a fake `~zod`. This is used only for development.
To use, run vere with the `-F` option (and the `-I ~zod` option for a fake
`~zod`).
###`++sufi`, domestic host
```
++ sufi :: domestic host
$: hoy=(list ship) :: hierarchy
val=wund :: private keys
law=will :: server will
seh=(map hand ,[p=ship q=@da]) :: key cache
hoc=(map ship dore) :: neighborhood
== ::
```
This is the security state of a domestic server.
`hoy` is a list of the ships directly above us in the hierarchy of ships. For
example, for `~hoclur-bicrel`, this would be `~tasruc` and `~tug`. See
`++sein`.
`val` is a list of our private keys.
`law` is our certificate, which is a list of the XXX
`seh`
`hoc` is a map of ships to `++dore`. The stores all the security informatoin
about foreign ships. The keys to this map are the neighbors (ships we have
been in contact with) of this domestic server.
###`++wund`, private keys
```
++ wund (list ,[p=life q=ring r=acru]) :: mace in action
```
This is a list of our own private keys, indexed by life. The key itself is
the `++ring`, and the `++acru` is the encryption engine. We generate the
`++acru` from the private key by calling `++weur`. Thus, we can at any time
regenerate our `++wund` from a `++mace`. The current crypto is at the head of
the list and can be accessed with
`++sen:as:go`.
###`++ring`, private key
```
++ ring ,@ :: private key
```
This is a private key. The first byte is reserved to identify the type of
cryptography. Lower-case means public key, upper-case means public key, and
the letter identifies which `++acru` to use.
###`++pass`, public key
```
++ pass ,@ :: public key
```
This is a public key. The first byte is reserved to identify the type of
cryptography. Lower-case means public key, upper-case means public key, and
the letter identifies which `++acru` to use.
###`++mace`, private secrets
```
++ mace (list ,[p=life q=ring]) :: private secrets
```
This is a list of the our private keys, indexed by life. From this we can
generate a `++wund` for actual use.
###`++skin`, encoding stem
```
++ skin ?(%none %open %fast %full) :: encoding stem
```
This defines the type of encryption used for each message. `%none` refers
to messages sent in the clear, `%open` refers to signed messages, `%full`
refers to sealed messages, and `%fast` refers to symmetrically encrypted
messages. See `++acru` for details.
###`++acru`, asymmetric cryptosuite
```
++ acru :: asym cryptosuite
$_ ^? |% :: opaque object
++ as ^? :: asym ops
|% ++ seal |=([a=pass b=@ c=@] _@) :: encrypt to a
++ sign |=([a=@ b=@] _@) :: certify as us
++ sure |=([a=@ b=@] *(unit ,@)) :: authenticate from us
++ tear |= [a=pass b=@] :: accept from a
*(unit ,[p=@ q=@]) ::
-- ::
++ de |+([a=@ b=@] *(unit ,@)) :: symmetric de, soft
++ dy |+([a=@ b=@] _@) :: symmetric de, hard
++ en |+([a=@ b=@] _@) :: symmetric en
++ ex ^? :: export
|% ++ fig _@uvH :: fingerprint
++ pac _@uvG :: default passcode
++ pub *pass :: public key
++ sec *ring :: private key
--
++ nu ^? :: reconstructors
|% ++ pit |=([a=@ b=@] ^?(..nu)) :: from [width seed]
++ nol |=(a=@ ^?(..nu)) :: from naked ring
++ com |=(a=@ ^?(..nu)) :: from naked pass
--
--
```
This is an opaque interface for a general asymmetric cryptosuite. Any form
of asymmetric cryptography can be dropped in to be used instead of the default.
Right now, there are two cryptosuites, `++crua`, which is your standard RSA,
and `++crub`, which is elliptic curve crypto but is mostly stubbed out at the
moment.
####`++as:acru`, asymmetric operations
```
++ as ^? :: asym ops
|% ++ seal |=([a=pass b=@ c=@] _@) :: encrypt to a
++ sign |=([a=@ b=@] _@) :: certify as us
++ sure |=([a=@ b=@] *(unit ,@)) :: authenticate from us
++ tear |= [a=pass b=@] :: accept from a
*(unit ,[p=@ q=@]) ::
-- ::
```
This is the core that defines the standard asymmetric cryptography
operations.
`++seal:as:acru` allows us to send a message encrypted with someone's public
key so that only they may read it. If Alice seals a message with Bob's public
key, then she can be sure that Bob is the only one who can read it. This is
associated with the `++skin` `%full`.
`++sign:as:acru` allows us to sign a message with our private key so that
others can verify that we sent the message. If Alice signs a message with her
private key, then Bob can verify with her public key that it was indeed Alice
who sent it. This is associated with the `++skin` `%open`.
`++sure:as:acru` is the dual to `++sign:as:acru`. It allows us to verify that
a message we have received is indeed from the claimed sender. If Alice sends a
message with her private key, then Bob can use this arm to verify that it was
indeed Alice who sent it. This is associated with the `++skin` `%open`.
`++tear:as:acru` is the dual to `++seal:as:acru`. It allows us to read a
message that we can be sure is only read by us. If Alice seals a message with
Bob's public key, then Bob can use this arm to read it. This is associated
with the `++skin` `%full`.
####`++de:acru`, `++dy:acru`, and `++en:acru`, symmetric encryption/decryption
```
++ de |+([a=@ b=@] *(unit ,@)) :: symmetric de, soft
++ dy |+([a=@ b=@] _@) :: symmetric de, hard
++ en |+([a=@ b=@] _@) :: symmetric en
```
Symmetric encryption is associated with the `++skin` `%fast`.
`++de:acru` decrypts a message with a symmetric key, returning `~` on failure
and `[~ u=data]` on success.
`++dy:acru` decrypts a message with a symmetric key, crashing on failure. This
should almost always be defined as, and should always be semantically
equivalent to, `(need (de a b))`.
`++en:acru` encrypts a message with a symmetric key.
####`++ex:acru`, exporting data
```
++ ex ^? :: export
|% ++ fig _@uvH :: fingerprint
++ pac _@uvG :: default passcode
++ pub *pass :: public key
++ sec *ring :: private key
--
```
`++fig:ex:acru` is our fingerprint, usually a hash of our public key. This is
used, for example, in `++zeno`, where every carrier owner's fingerprint is
stored so that we can ensure that carriers are indeed owned by their owners
`++pac:ex:acru` is our default passcode, which is unused at present.
`++pub:ex:acru` is the `++pass` form of our public key.
`++sec:ex:acru` is the `++ring` form of our private key.
####`++nu:acru`, reconstructors
```
++ nu ^? :: reconstructors
|% ++ pit |=([a=@ b=@] ^?(..nu)) :: from [width seed]
++ nol |=(a=@ ^?(..nu)) :: from naked ring
++ com |=(a=@ ^?(..nu)) :: from naked pass
--
```
These arms allow us to reconstruct a `++acru` from basic data.
`++pit:nu:acru` constructs a `++acru` from the width of our intended key and
seed entropy. This is usually used in the initial construction of the
`++acru`.
`++nol:nu:acru` constructs a `++acru` from a "naked ring", meaning a `++ring`
without the initial byte identifying the type of crypto. There is often a
helper arm that that wraps this; see `++weur` for `++crua` and `++wear` for
`++crub`.
`++com:nu:acru` constructs a `++acru` from a "naked pass", meaning a `++ring`
without the initial byte identifying the type of crypto. There is often a
helper arm that that wraps this; see `++haul` for `++crua` and `++hail` for
`++crub`.
###`++will`, certificate
```
++ will (list deed) :: certificate
```
This is a list of deeds associated with the current ship. There should be
an item in this list for every ship from this point up in the hierarchy times
the number of lives that each ship has had. For example, ~hoclur-bicrel may
have a will with three items: one for itself, one for ~tasruc (who issued
~hoclur-bicrel's deed) and one for ~tug (who issued ~tasruc's deed).
###`++deed`, identity
```
++ deed ,[p=@ q=step r=?] :: sig, stage, fake?
```
`p` is the signature of a particular deed, which is a signed copy of `q`.
`q` is the stage in the identity.
`r` is true if we're working on a fake network, where we don't check that the
carrier fingerprints are correct. This allows us to create fake networks for
development without interfering with the real network.
###`++step`, identity stage
```
++ step ,[p=bray q=gens r=pass] :: identity stage
```
This is a single stage in our identity. Thus, this is specific to a single
life in a single ship. Everything in here may change between lives.
`p`
`q`
`r` is the public key for this stage in the identity.
###`++bray`
```
++ bray ,[p=life q=(unit life) r=ship s=@da] :: our parent us now
```
XXX
###`++gens`, general identity
```
++ gens ,[p=lang q=gcos] :: general identity
```
`p` is the IETF language code for the preferred language of this identity.
This is unused at the moment, but in the future text should be localized based
on this.
`q` is the description of the ship.
###`++gcos`, identity description
```
++ gcos :: id description
$% [%czar ~] :: 8-bit ship
[%duke p=what] :: 32-bit ship
[%earl p=@t] :: 64-bit ship
[%king p=@t] :: 16-bit ship
[%pawn p=(unit ,@t)] :: 128-bit ship
== ::
```
This is the description of the identity of a ship. Most types of identity have
a `@t` field, which is their human-readable name. The identity of a `%duke` is
more involved.
A `%czar`, a carrier, is a ship with an 8-bit address. Thus, there are only
256 carriers. These are at the top of the namespace hierarchy, and the
fingerprint of each carrier is stored in `++zeno`. These are the "senators" of
Urbit.
A `%king`, a cruiser, is a ship with a 16-bit address. Thus, there are 65,536
cruisers. Each carrier may issue 256 cruisers. These are the infrastructure
of Urbit.
A `%duke`, a destroyer, is a ship with a 32-bit address. Thus, there are
4,294,967,296 destroyers. Each cruiser may issue 65,536 cruisers. These are
the individuals of Urbit.
A `%earl`, a yacht, is a ship with a 64-bit address. Thus, there are
18,446,744,073,709,551,616 yachts. Each destroyer may issue 4,294,967,296
yachts. These are the devices of Urbit.
A `%pawn`, a submarine, is a ship with a 128-bit address. Thus, there are a
lot of submarines. The chance of random name collision is negligible, so
submarines are not issued by any ship. They must simply assert their presence,
and they are all considered children of ~zod. This is the underworld of Urbit,
where anonymity reigns supreme.
###`++what`, logical destroyer identity
```
++ what :: logical identity
$% [%anon ~] :: anonymous
[%lady p=whom] :: female person ()
[%lord p=whom] :: male person []
[%punk p=sect q=@t] :: opaque handle ""
== ::
```
This is the logical identity of a destroyer.
A `%anon` is a completely anonymous destroyer. The difference between this and
a submarine is that a submarine is ephemeral while a `%anon` destroyer is not.
Thus, we may not know who ~hoclur-bicrel is, but we do know that it's always
the same person.
A `%lady` is a female person. The name used here should be a real name.
A `%lord` is a male person. The name used here should be a real name.
A `%punk` is a person who is identified only by a handle.
###`++whom`, real person
```
++ whom ,[p=@ud q=govt r=sect s=name] :: year/govt/id
```
Ths is the information associated with a real person. It is mostly information
that could be observed with the briefest of interactions.
`p` is the birth year.
`q` is the location of a user, usually of the form "country/zip".
`r` is the sect of the user.
`s` is the real name of the person.
###`++govt`
```
++ govt path :: country/postcode
```
This is the location of the user, usually of the form "country/zip".
###`++sect`
```
++ sect ?(%black %blue %red %orange %white) :: banner
```
XXX
###`++name`
```
++ name ,[p=@t q=(unit ,@t) r=(unit ,@t) s=@t] :: first mid/nick last
```
This is the given name, possible middle name/initial, possible nickname, and
surname of a user.
packet format
-------------
`++go`, PKI engine
------------------
###`++as`, per server
####`++born`, register user
#####`++lax`, per client
`++pu`, packet pump
-------------------
`++am`, protocol engine
-----------------------
###`++um`, per server
####`++ho`, per friend
#####`++la`, per packet
protocol vane
-------------
#Batz
Coming soon
#Dill
Coming soon
#Clay
Clay
====
Clay is our filesystem.
data models
-----------
###`++raft`, formal state
```
++ raft :: filesystem
$: fat=(map ship room) :: domestic
hoy=(map ship rung) :: foreign
ran=rang :: hashes
== ::
```
This is the state of our vane. Anything that must be remembered between calls
to clay must be stored in this state.
`fat` is the set of domestic servers. This stores all the information that is
specfic to a particular ship on this pier. The keys to this map are the ships
on the current pier.
`hoy` is the set of foreign servers that we know anything about. This stores
all the information that is specific to a particular foreign ship. The keys to
this map are all the ships whose filesystems we have attempted to access
through clay.
`ran` is the store of all commits and deltas, keyed by hash. The is where all
the "real" data we know is stored; the rest is "just bookkeeping".
###`++room`, filesystem per domestic ship
```
++ room :: fs per ship
$: hun=duct :: terminal duct
hez=(unit duct) :: sync duch
dos=(map desk dojo) :: native desk
== ::
```
This is the representation of the filesystem of a ship on our pier.
`hun` is the duct that we use to send messages to dill to display notifications
of filesystem changes. Only `%note` gifts should be produced along this duct.
This is set by the `%init` kiss.
`hez`, if present, is the duct we use to send sync messages to unix so that
they end up in the pier unix directory. Only `%ergo` gifts should be producd
along this duct. This is set by `%into` and `%invo` gifts.
`dos` is a well-known operating system released in 1981. It is also the set of
desks on this ship, mapped to their data.
###`++desk`, filesystem branch
```
++ desk ,@tas :: ship desk case spur
```
This is the name of a branch of the filesystem. The default desks are "arvo",
"main", and "try". More may be created by simply referencing them. Desks have
independent histories and states, and they may be merged into each other.
###`++dojo`, domestic desk state
```
++ dojo ,[p=cult q=dome] :: domestic desk state
```
This is the all the data that is specific to a particular desk on a domestic
ship. `p` is the set of subscribers to this desk and `q` is the data in the
desk.
###`++cult`, subscriptions
```
++ cult (map duct rave) :: subscriptions
```
This is the set of subscriptions to a particular desk. The keys are the ducts
from where the subscriptions requests came. The results will be produced along
these ducts. The values are a description of the requested information.
###`++rave`, general subscription request
```
++ rave :: general request
$% [& p=mood] :: single request
[| p=moat] :: change range
== ::
```
This represents a subscription request for a desk. The request can be for
either a single item in the desk or else for a range of changes on the desk.
###`++mood`, single subscription request
```
++ mood ,[p=care q=case r=path] :: request in desk
```
This represents a request for the state of the desk at a particular commit,
specfied by `q`. `p` specifies what kind of information is desired, and `r`
specifies the path we are requesting.
###`++moat`, range subscription request
```
++ moat ,[p=case q=case] :: change range
```
This represents a request for all changes between `p` and `q`. Note that there
is currently no way to request to be notified only on changes to particular
paths in the filesystem. You must subscribe to the entire desk.
###`++care`, clay submode
```
++ care ?(%u %v %w %x %y %z) :: clay submode
```
This specifies what type of information is requested in a subscription or a
scry.
`%u` requests the `++rang` at the current moment. Because this information is
not stored for any moment other than the present, we crash if the `++case` is
not a `%da` for now.
`%v` requests the `++dome` at the specified commit.
`%w` requests the current revsion number of the desk.
`%x` requests the file at a specified path at the specified commit. If there
is no node at that path or if the node has no contents (that is, if `q:ankh` is
null), then this produces null.
`%y` requests a `++arch` of the specfied commit at the specified path.
`%z` requests the `++ankh` of the specified commit at the specfied path.
###`++arch`, shallow filesystem node
```
++ arch ,[p=@uvI q=(unit ,@uvI) r=(map ,@ta ,~)] :: fundamental node
```
This is analogous to `++ankh` except that the we have neither our contents nor
the ankhs of our children. The other fields are exactly the same, so `p` is a
hash of the associated ankh, `u.q`, if it exists, is a hash of the contents of
this node, and the keys of `r` are the names of our children. `r` is a map to
null rather than a set so that the ordering of the map will be equivalent to
that of `r:ankh`, allowing efficient conversion.
###`++case`, specifying a commit
```
++ case :: ship desk case spur
$% [%da p=@da] :: date
[%tas p=@tas] :: label
[%ud p=@ud] :: number
== ::
```
A commit can be referred to in three ways: `%da` refers to the commit that was
at the head on date `p`, `%tas` refers to the commit labeled `p`, and `%ud`
refers to the commit numbered `p`. Note that since these all can be reduced
down to a `%ud`, only numbered commits may be referenced with a `++case`.
###`++dome`, desk data
```
++ dome :: project state
$: ang=agon :: pedigree
ank=ankh :: state
let=@ud :: top id
hit=(map ,@ud tako) :: changes by id
lab=(map ,@tas ,@ud) :: labels
== ::
```
This is the data that is actually stored in a desk.
`ang` is unused and should be removed.
`ank` is the current state of the desk. Thus, it is the state of the
filesystem at revison `let`. The head of a desk is always a numbered commit.
`let` is the number of the most recently numbered commit. This is also the
total number of numbered commits.
`hit` is a map of numerical ids to hashes of commits. These hashes are mapped
into their associated commits in `hut:rang`. In general, the keys of this map
are exactly the numbers from 1 to `let`, with no gaps. Of course, when there
are no numbered commits, `let` is 0, so `hit` is null. Additionally, each of
the commits is an ancestor of every commit numbered greater than this one.
Thus, each is a descendant of every commit numbered less than this one. Since
it is true that the date in each commit (`t:yaki`) is no earlier than that of
each of its parents, the numbered commits are totally ordered in the same way
by both pedigree and date. Of course, not every commit is numbered. If that
sounds too complicated to you, don't worry about it. It basically behaves
exactly as you would expect.
`lab` is a map of textual labels to numbered commits. Note that labels can
only be applied to numbered commits. Labels must be unique across a desk.
###`++ankh`, filesystem node
```
++ ankh :: fs node (new)
$: p=cash :: recursive hash
q=(unit ,[p=cash q=*]) :: file
r=(map ,@ta ankh) :: folders
== ::
```
This is a single node in the filesystem. This may be file or a directory or
both. In earth filesystems, a node is a file xor a directory. On mars, we're
inclusive, so a node is a file ior a directory.
`p` is a recursive hash that depends on the contents of the this file or
directory and on any children.
`q` is the contents of this file, if any. `p.q` is a hash of the contents
while `q.q` is the data itself.
`r` is the set of children of this node. In the case of a pure file, this is
empty. The keys are the names of the children and the values are, recursively,
the nodes themselves.
###`++cash`, ankh hash
```
++ cash ,@uvH :: ankh hash
```
This is a 128-bit hash of an ankh. These are mostly stored within ankhs
themselves, and they are used to check for changes in possibly-deep
hierarchies.
###`++rung`, filesystem per neighbor ship
```
++ rung $: rus=(map desk rede) :: neighbor desks
== ::
```
This is the filesystem of a neighbor ship. The keys to this map are all the
desks we know about on their ship.
###`++rede`, desk state
```
++ rede :: universal project
$: lim=@da :: complete to
qyx=cult :: subscribers
ref=(unit rind) :: outgoing requests
dom=dome :: revision state
== ::
```
This is our knowledge of the state of a desk, either foreign or domestic.
`lim` is the date of the last full update. We only respond to requests for
stuff before this time.
`qyx` is the list of subscribers to this desk. For domestic desks, this is
simply `p:dojo`, all subscribers to the desk, while in foreign desks this is
all the subscribers from our ship to the foreign desk.
`ref` is the request manager for the desk.
`dom` is the actual data in the desk.
###`++rind`, request manager
```
++ rind :: request manager
$: nix=@ud :: request index
bom=(map ,@ud ,[p=duct q=rave]) :: outstanding
fod=(map duct ,@ud) :: current requests
haw=(map mood (unit)) :: simple cache
== ::
```
This is the request manager for a desk.
`nix` is one more than the index of the most recent request. Thus, it is the
next available request number.
`bom` is the set of outstanding requests. The keys of this map are some subset
of the numbers between 0 and one less than `nix`. The members of the map are
exactly those requests that have not yet been fully satisfied.
`fod` is the same set as `bom`, but from a different perspective. In
particular, the values of `fod` are the same as the values of `bom`, and the
`p` out of the values of `bom` are the same as the keys of `fod`. Thus, we can
map ducts to their associated request number and `++rave`, and we can map
numbers to their associated duct and `++rave`.
`haw` is a map from simple requests to their values. This acts as a cache for
requests that have already been made. Thus, the second request for a
particular `++mood` is nearly instantaneous.
###`++rang`, data store
```
++ rang $: hut=(map tako yaki) ::
lat=(map lobe blob) ::
== ::
```
This is a set of data keyed by hash. Thus, this is where the "real" data is
stored, but it is only meaningful if we know the hash of what we're looking
for.
`hut` is a map from hashes to commits. We often get the hashes from
`hit:dome`, which keys them by logical id. Not every commit has an id.
`lat` is a map from hashes to the actual data. We often get the hashes from a
`++yaki`, a commit, which references this map to get the data. There is no
`++blob` in any `++yaki`. They are only accessible through this map.
###`++tako`, commit reference
```
++ tako ,@ :: yaki ref
```
This is a hash of a `++yaki`, a commit. These are most notably used as the
keys in `hut:rang`, where they are associated with the actual `++yaki`, and as
the values in `hit:dome`, where sequential ids are associated with these.
###`++yaki`, commit
```
++ yaki ,[p=(list tako) q=(map path lobe) r=tako t=@da] :: commit
```
This is a single commit.
`p` is a list of the hashes of the parents of this commit. In most cases, this
will be a single commit, but in a merge there may be more parents. In theory,
there may be an arbitrary number of parents, but in practice merges have
exactly two parents. This may change in the future. For commit 1, there is no
parent.
`q` is a map of the paths on a desk to the data at that location. If you
understand what a `++lobe` and a `++blob` is, then the type signature here
tells the whole story.
`r` is the hash associated with this commit.
`t` is the date at which this commit was made.
###`++lobe`, data reference
```
++ lobe ,@ :: blob ref
```
This is a hash of a `++blob`. These are most notably used in `lat:rang`, where
they are associated with the actual `++blob`, and as the values in `q:yaki`,
where paths are associated with their data in a commit.
###`++blob`, data
```
++ blob $% [%delta p=lobe q=lobe r=udon] :: delta on q
[%direct p=lobe q=* r=umph] ::
[%indirect p=lobe q=* r=udon s=lobe] ::
== ::
```
This is a node of data. In every case, `p` is the hash of the blob.
`%delta` is the case where we define the data by a delta on other data. In
practice, the other data is always the previous commit, but nothing depends on
this. `q` is the hash of the parent blob, and `r` is the delta.
`%direct` is the case where we simply have the data directly. `q` is the data
itself, and `r` is any preprocessing instructions. These almost always come
from the creation of a file.
`%indirect` is both of the preceding cases at once. `q` is the direct data,
`r` is the delta, and `s` is the parent blob. It should always be the case
that applying `r` to `s` gives the same data as `q` directly (with the
prepreprocessor instructions in `p.r`). This exists purely for performance
reasons. This is unused, at the moment, but in general these should be created
when there are a long line of changes so that we do not have to traverse the
delta chain back to the creation of the file.
###`++udon`, abstract delta
```
++ udon :: abstract delta
$: p=umph :: preprocessor
$= q :: patch
$% [%a p=* q=*] :: trivial replace
[%b p=udal] :: atomic indel
[%c p=(urge)] :: list indel
[%d p=upas q=upas] :: tree edit
== ::
== ::
```
This is an abstract change to a file. This is a superset of what would
normally be called diffs. Diffs usually refer to changes in lines of text
while we have the ability to do more interesting deltas on arbitrary data
structures.
`p` is any preprocessor instructions.
`%a` refers to the trival delta of a complete replace of old data with new
data.
`%b` refers to changes in an opaque atom on the block level. This has very
limited usefulness, and is not used at the moment.
`%c` refers to changes in a list of data. This is often lines of text, which
is your classic diff. We, however, will work on any list of data.
`%d` refers to changes in a tree of data. This is general enough to describe
changes to any hoon noun, but often more special-purpose delta should be
created for different content types. This is not used at the moment, and may
in fact be unimplemented.
###`++urge`, list change
```
++ urge |*(a=_,* (list (unce a))) :: list change
```
This is a parametrized type for list changes. For example, `(urge ,@t)` is a
list change for lines of text.
###`++unce`, change part of a list.
```
++ unce |* a=_,* :: change part
$% [%& p=@ud] :: skip[copy]
[%| p=(list a) q=(list a)] :: p -> q[chunk]
== ::
```
This is a single change in a list of elements of type `a`. For example, `(unce ,@t)` is
a single change in a lines of text.
`%&` means the next `p` lines are unchanged.
`%|` means the lines `p` have changed to `q`.
###`++umph`, preprocessing information
```
++ umph :: change filter
$| $? %a :: no filter
%b :: jamfile
%c :: LF text
== ::
$% [%d p=@ud] :: blocklist
== ::
```
This space intentionally left undocumented. This stuff will change once we get
a well-typed clay.
###`++upas`, tree change
```
++ upas :: tree change (%d)
$& [p=upas q=upas] :: cell
$% [%0 p=axis] :: copy old
[%1 p=*] :: insert new
[%2 p=axis q=udon] :: mutate!
== ::
```
This space intentionally left undocumented. This stuff is not known to work,
and will likely change when we get a well-typed clay. Also, this is not a
complicated type; it is not difficult to work out the meaning.
#Eyre
Coming soon
#Ford
Coming soon
#Gall
Coming soon

View File

@ -1,391 +0,0 @@
---
layout: post
category: blog
author: cgy
title: Welcome to Urbit
date: 2013-9-24 15:00
---
But wait - what the hell is Urbit?
One of Urbit's problems is that we don't exactly have a word for
what Urbit is. If there is such a word, it somehow means both
"operating system" and "network protocol," while somehow also
implying "functional" and "deterministic."
Not only is there no such word, it's not even clear there
_should_ be one. And if there was, could we even hear it?
As Wittgenstein said: if a lion could talk, we would not
understand him. But heck, let's try anyway.
As a network protocol, we could call Urbit a "seven-layer
protocol" - that is, a protocol that specifies the complete
semantics of the general-purpose computer that processes it. As
any decent IETF ninja will tell you, this is a very bad idea for
all sorts of extremely obvious reasons.
And from the OS perspective, Urbit is yet another slice of
userspace crap with the temerity to call itself an "operating
system." Urbit is not actually an OS in the bare-metal sense.
It's a VM that runs on Linux or OS X. Someday it might be so
daring as to descend to Xen. Urbit has no interest at all in
drivers, peripherals, etc. It is just a UDP transceiver in the
cloud. Worst of all - Urbit is not even preemptive. A poser!
(Actually all the real work, as in node, is done by libuv.)
Moreover, this VM is formally isolated from your actual OS. And
everything that uses it. Nothing in Urbit can request system
services or call existing libraries. So Urbit is not only badly
designed and fraudulently hyped. It's also profoundly useless.
Well... your browser has been reasonably successful with this
restriction. But your browser was never designed to be any kind
of OS. To the extent that it's sort of become an OS, it is
specialized for the very undemanding job of being a client. A
general-purpose client, which is neat. But not a general-purpose
server - which is a much harder problem.
A general-purpose server is a slab of code that feels totally
confident when faced with the problem of storing _all your
personal and/or corporate data_, across _arbitrary functional
upgrades_, for _all time forever_, while efficiently executing
and managing _any useful program or programs, transient or
persistent_. Yeah, that's a server OS.
So, conventionally, this industrial strength slab of code is
written with conventional OS techniques involving (a) bare metal
and (b) like, semaphores and shit. The kernel alone is like 12
million lines of code. Not that a browser is any much smaller.
And so, 20th-century network computing is the world's most
beautiful wire, between two dumpsters of shit spaghetti. Two
_completely different_ dumpsters. It turns out that with a big
enough dumpster of shit spaghetti, you can feed the world. And
why not two? Incompatibility creates jobs, you know.
Oh, and you can have a client without an identity. But you can't
have a _server_ without an identity. So Urbit has to solve _that_
problem. Unless it's just going to be a Web server. (Urbit is
actually a perfectly decent Web server.) Unless it has an actual
identity model, your general-purpose server - presumably a
_network_ server - has no network effect. No identity, no
network. No network, who the hell cares?
And unlike your ordinary, common or garden Web application
server, Urbit does need that network effect. You see, it's not
even our own work. It's something we found. On an unmarked USB
stick by the side of the road three kilometers from Area 51.
We think it's probably still human engineering. First of all,
there are no aliens. Second, Urbit uses Unicode. Do the aliens
have Unicode? Or SHA-256? _Where did Unicode come from,
anyway?_ And at the I/O level, we see UDP, HTTP, etc. The
command line does Emacs keys. At the very least, someone on
Earth (or at least _from_ Earth) has done some porting.
But other than these cosmetic details, there's not much sign of a
connection to ordinary Earth computing. For instance, Urbit
isn't written in any of our Earth languages. It is written in
something called Hoon, which is a strict, higher-order, typed
functional language - but has nothing in else in common with
other Earth languages, functional or not. Hoon does not even
use standard PL theory. And its syntax is just as alien, although
at least it confines itself to the ASCII plane. (And you
probably thought the "A" in "ASCII" stood for "American.")
Worse yet, Hoon is not written in anything normal. It's written
in Hoon. To be precise - the Hoon compiler compiles itself to a
much simpler automaton, Nock. Besides machine language itself,
and its various bytecode homages, there are three simple formal
models of computing - Turing machines, lambda calculus, and Chuck
Moore. There is also a fourth which no one has ever found useful
for anything: combinators. Nock is a combinator automaton.
While nowhere near the simplest such automaton known, and
certainly not of any _theoretical_ interest, Nock is so stupid
that if you gzip the spec, it's only 374 bytes. Nock's only
arithmetic operation is increment. So decrement is an `O(n)`,
operation; add is `O(m*n)`...
Clearly not a practical system. Even if this... _thing_... that
sits on top of it was in any way, shape or form remotely sane.
So why not try it? Heck, why not at least check it out? Strange
and foolish dreamers may hie themselves to teh github:
[https://github.com/urbit](https://github.com/urbit)
Various people have built and run Urbit on Ubuntu, Homebrew and
MacPorts. It's possible that all these people were stone cold
ninjas. Urbit is a pretty cool toy, I think, if you're a ninja.
Unfortunately it is not really ready for ordinary rice farmers.
If you can't resolve build dependencies by hand, we're sorry.
Urbit is probably not yet ready for you.
Where should you run Urbit? Ideally, in the cloud. Urbit can
punch a decent NAT hole. It doesn't like to, though - what does?
Bear in mind that your Urbit instance is a single-level store -
your state is saved as a log and a checkpoint (as in Redis -
except that Redis is both fast and reliable). This will work
much better on server hardware. That said, there are plenty of
good reasons to keep your servers in the closet with the plants.
Next, you need a ticket. Or not. You at least need to decide
whether you want a ticket or not. Actually, the answer is
simple. You do want one. But you don't need one - not yet.
Because Urbit, the OS, never depends on Earth semantics, it needs
its own network protocol - Ames. Ames is a P2P protocol with its
own public-key infrastructure. (Ames is encrypted, but the
current cryptosuite, suite A, is worthless snakeoil. Don't trust
it with your Trader Joe receipts.) Ames runs over UDP, and pays
as little attention to IP routing as possible. Needless to say,
Urbit does not use the DNS at all.
To sling packets on Ames, you need an identity. Needless to say,
there are a lot of ways to do distributed cryptographic identity,
all of them bad.
The general reason all PKIs suck is called "Zooko's Triangle."
Your identities can be distributed, secure, or human-meaningful -
pick any two. There is no way to solve Zooko's Triangle. The
only way to attack it is to compromise on at least one vertex.
To see how Urbit works, let's start with a standard solution. An
Urbit identity is called a "ship." You can launch your own ship
by generating a 2048-bit RSA key and hashing it to a 128-bit
fingerprint, which is your identity. This trivial design is the
basis of all distributed, secure PKIs.
Unfortunately, an identity should be above all a _name_. A
128-bit fingerprint looks like this:
316.931.986.049.624.498.975.974.005.376.666.572.699
which isn't a name. For no better reason than the fact that,
unless you're an autistic savant, you are basically a monkey with
an overgrown monkey brain. A name is something that fits in a
register. Your monkey brain doesn't have 128-bit registers.
Suppose we could use 64-bit fingerprints? At first this seems
even less promising. First of all, your monkey brain doesn't
have 64-bit registers, either. Second, at 64 bits, collisions
are already a real problem.
But we can solve both these problems. Your monkey brain doesn't
have 64-bit registers. But anything that lets us do 64-bit
identities might stretch down to 32 bits. And at 64 or 32 bits,
we can't assign identities by random hash. So we'll have to find
another way of distributing them.
A 32-bit identity - or even a 16-bit identity - will still never
be human-meaningful. Perhaps we can make it human-memorable.
Meaningful addresses are a nice feature in a social network, but
memorable addresses are essential. And if we have memorable
addresses, perhaps we can build meaningful names on top.
The classic example of a memorable address is a street address.
Your street address identifies you precisely, using a name. You
have this great brain hardware for remembering names. But what
does the name of your street _mean_? Nothing at all, usually.
And even if it does mean something, by accident or design, that
meaning has nothing at all to do with you. (One of us grew up in
Columbia, Maryland, which has street names like "Greek Boy
Place.")
So all we need is a simple, tasteful, memorable way to remember
32 bits - and we're on our way to approximating the Triangle.
Decimal notation is the worst way of remembering a 32-bit number.
IP notation is a little better. Urbit has a typically crude
approach: we map every byte to a CVC phoneme, making names like:
~tasfyn-partyv
~sivbud-barnel
~tomsyt-balsen
These strings, while quite meaningless, are no less memorable
than real human names in many a language. Moreover, they form a
language of their own, and become more memorable as you use them.
And there are 4 billion of them, which (as we'll see) is almost
exactly the right number.
But how do we distribute them? One obvious solution is a proof
of work scheme, as in Bitcoin. Coordinating a global
proof-of-work scheme is quite nontrivial, however. Also, there
is a second reason to compromise 100% decentralization: packet
routing. It might be possible to use a blockchain as a global
routing table. It would take some thinking about.
Furthermore, there's a clue here that the Bitcoin approach just
isn't getting. The limited subspace of short names, within the
general space of 128-bit names, is essentially _real estate_.
There is absolutely no reason, moral or practical, to give this
real estate away for free to people whose only contribution is
generating CO2 on their GPUs. Mining is not in any way a
productive activity.
Rather, initially, this real estate belongs to Urbit itself. If
Urbit has value, its real estate has value. If Urbit has no
value, its so-called real estate is a bunch of worthless bits.
Therefore, any value in the real estate can, should, and will be
used to bootstrap the system from an economic perspective. Ie,
it belongs to and will be captured by Urbit's developers and/or
early adopters. If you find this morally wrong, sorry. You're
probably some kind of a communist.
But because Urbit is a free republican society - not (ahem) a
fascist corporate dictatorship like Google, Facebook or Twitter -
a crucial aspect of launching or transferring a ship is that the
decision is irreversible.
As the master of an Urbit ship, your informal title is
cryptographic and _allodial_ - no one, not the government and
certainly not us, can challenge it. Unless the attacker can
steal your secrets. In which case, of course, she might as well
be you. That's like Bitcoin too.
If Bitcoin is money, Urbit is land. (Floating land is still
land, if there's a limited amount of it.) You own both in the
same way, by proving you can keep a secret. A Bitcoin is not
useful for anything, except selling to a greater fool. (We're
just kidding - we're huge Bitcoin fans.) But an Urbit ship is
directly useful, so long as Urbit itself is useful.
You fill your Bitcoin wallet either by creating new coins, or
buying old ones from Satoshi and his cronies. You build your
Urbit fleet by buying ships from us and our cronies. (Don't ask
what we had to do to get them from the aliens. Those aliens are
into a lot of strange shit, man.) Ships are transferable, but
Urbit is not designed to be a digital currency. Transaction
overhead is artificially high. Again, as in real estate.
Urbit at present is empty and worthless. So 32-bit ships -
destroyers - are $0. Launch a 128-bit ship (a submarine) and
ask me, ~tasfyn-partyv, for one. We'll send you one, two, or a
dozen. You can be as anonymous as you want, if you're polite.
But, if the network lives, a destroyer price will develop. It
will be very low at first, but not zero. Urbit is designed to be
free as in speech. It's not designed to be free as in beer.
How, cryptographically, are cloud ships distributed? And how are
packets routed? The answer is the same - the prefix hierarchy.
In Urbit's naval terminology, ships above 64 bits are
"submarines." 64-bit ships are "yachts." 32-bit, "destroyers."
16-bit, "cruisers." 8-bit, "carriers." This also resembles a
feudal hierarchy, so it comes with a feudal terminology. There
are 256 imperial carriers, 65.280 royal cruisers, 4.294.901.760
rebel destroyers, uncounted scads of private yachts, and more or
less infinitely many rogue submarines.
Every ship but a submarine has a "will," or certificate chain - a
linked list of "deeds." The first deed in this list is signed by
the ship's hierarchical prefix, or "flagship." Mere submarine are
independent; carriers create cruisers; cruisers create
destroyers; destroyers create yachts.
A submarine is the fingerprint of its own public key; a carrier's
fingerprint is predefined in the kernel. Anyone can create any
number of 128-bit submarines, whose free and independent
society the 64-bit naval hierarchy cannot interfere with. And
of course, since Urbit is (a) in the public domain and (b) not
patented, anyone can fork Urbit and change the carrier
fingerprints. Anyone can also create his own independent and
incompatible DNS, but efforts in this direction have not been
crowned with great success. In general, the easier it is
technically to fork open-source code or an open standard, the
less likely a fork is to actually happen.
An independent ship is an independent reputation. Your flagship
hierarchy, though it created your ship, has no control over it -
so your reputations are and should be separate. But there are
not 2^64 independent reputations in Urbit, only 2^32. Cruisers
have no control over the destroyers they create, but yachts have
no independence from the destroyers that created them.
The reason for this is simple - a destroyer corresponds not to a
person, but to any institution with its own independent
reputation. Yachts are for users, bots, or other sub-identities
of this institution. Each destroyer has 2^32 of them, which is,
of course, a lot.
How does independence work in practice? By pinning/TOFU. For
any deed number, or "life," the first instance is accepted. Thus
when changing secrets, perhaps to transfer a ship, the donor
signs a new deed created by the recipient. Once any ship sees
this deed, it will never accept another signed by the old owner.
Thus, a cruiser cannot sell the same new destroyer twice.
Similarly, deed 7 of ~tasfyn-partyv signs deed 8; but no ship
which has a deed 8 for ~tasfyn-partyv will either accept deed 7,
or any other purported deed 8 which deed 7 later signs.
Preventing a "double spend" thus depends on propagating the
latest deed. For this purpose, the ocean is divided into two
kinds of ships: friends and non-friends of the ship being
transferred. The ship has a list of its cryptographic partners,
or "neighbors," for which it holds a symmetric key and a routing
(IP) address.
The new owner sends the new deed to all the neighbors, thus
locking the old owner out of them. Future new friends will get
the new owner's will the same way they get the new owner's IP
address - by a lookup through the flagship hierarchy. Thus if we
update both neighbors and flagship, the old owner is locked out
and the new owner is locked in.
Technically, this is very much an imperfect procedure. It relies
on social trust to make it effective. For example, a malicious
seller could edit his neighbor list before delivering the ship.
You probably don't want to buy a used destroyer from someone you
distrust. Of course, the same is true of cars.
And more broadly, the naval hierarchy can and should provide
general administrative support. For one thing, the big ships
route your packets, at least when establishing contact with your
fellow destroyers.
So - do you want an Urbit destroyer? You know you do. Moreover,
they're free. Of course, if you are satisfied with a name that
looks like
~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted
you can keep swimming around Urbit in your submarine.
For a while, anyway. Because, besides memorability, there is
another reason for short names.
What's neat about short names is that there's a finite number of
them. This is not a bug, but a feature. Essentially, to borrow
the thinking of political scientist James Scott, a finite space
is _governable_. An infinite space is ungovernable.
If there are an infinite number of identities, there is no way
for anyone to distinguish between a new user and a banned user.
A reputation can be punished by destroying it, but anyone can
start again at zero. A parasite whose only reason to use the
network is to abuse it can keep coming back for more. An email
spammer will never run out of addresses to spam from.
IPv4 is a limited space, which almost but doesn't cure spam. The
problem is that IPv4 addresses are neither personal nor property,
so there is generally no easy way to punish a spammer as he
deserves through IP blacklisting. He is very unlikely to be in
any sense the owner of the IP address on his packets.
But if the email address and the IP address were the same thing,
and the present fuzzy economic relationship between the user of
an IP address were clear and simple, killing spam would become
easy. You spam from a destroyer; you go on a list of spammers;
no one will accept your unsolicited messages, ever.
You can get around this. You can buy a new destroyer. But the
thing is - it costs you *money*. You're not spamming for the
fun of it. If a destroyer costs a mere $1, the spam you send
from it needs to earn you $1.
This does not make it trivial for the forces of light to hunt you
down and render you into processed meat clippings. But it sure
as heck evens the game. Who will win? I guess we'll see.
So do you want an Urbit destroyer? Read the [documentation](/2013/11/18/urbit-is-easy-ch1.html) for instructions. Come on, they're free...

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-arvo
categories: tut
sort: 1
title: I - Basic Arvo
---
#Basic Arvo
These sections assume you've gotten Arvo up and running; if
not, [do so](/setup) first.

View File

@ -1,11 +1,6 @@
---
layout: tutpage
axis: doc-arvo
categories: tut
sort: 1
title: II - More Basic Arvo
---
#Local revision control
#More Basic Arvo
##Local revision control
Now you're ready to see the full power of this fully armed
and operational revision-controlled filesystem.
@ -146,7 +141,7 @@ But one could argue that it's just a party trick. But once your
revision control system and your scheduler are jammed into one
phone booth... why not jam networking in there too?
#Network revision control#
##Network revision control#
Arvo is the OS. Urbit is the network. In theory you could write
an Urbit client that wasn't Arvo, though it's hard to see why.

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,522 @@
# Urbit is a global functional filesystem.
The whole idea of a filesystem is that a file is a function of a
path. If it's an *immutable* filesystem like Urbit, the function
(though partial), once bound never changes.
This is kind of neat. What'd be really neat, though, is if the
namespace operator, rather than just returning one static file,
produced an immutable function on your whole filesystem. Or
better yet, on *all visible data in the world*.
It would also be ideal if the product of this function was not
just a binary blob, but a *typed data structure*. Even better -
the product might not be data, but code, or a mix of data and
code - a function, an object, a library, even an application.
What's basically irritating about the Web is that it's almost
this, but it's not this. The Web is almost a web-of-data, but
it's not a web-of-data. So it does not scale as a substrate for
complex higher-level communication patterns, and even for simple
use cases breaks frequently in production.
The road to a real *global functional filesystem* is perfectly
straightforward in principle. If a function of one path is
immutable, so is a function of two, or n. More generally, any
immutable function of a filename is a referentially transparent
name - a concept familiar in many functional languages.
We already put source files in immutable filesystems, which we
call "revision-control systems." To convert this design into a
functional filesystem, all we have to do is define a precise
functional semantics for these source files, including in that
semantics the power to read back into the namespace.
In principle this is quite straightforward. We need exactly two
components for our functional filesystem: an immutable filesystem
and a functional language. Neither are particularly exotic bits
of system software kit.
But they should be made to fit each other. Could you strap
github to Haskell and make a global functional filesystem? Well,
sort of. Certainly in principle. You could also strap a car
engine to a mountain bike, and make a motorcycle. It wouldn't be
a very good motorcycle.
Urbit is not actually a very good motorcycle. Not yet, anyway.
It is a motorcycle, though. Let's take it for a spin.
# Get your urbit.
First, grab yourself a "destroyer" at urbit.org. You'll come
away with an phonemic urbit name, or "ship" - mine is
`~tasfyn-partyv`. Let's say `~lacser-dovlet` is yours.
As the length of the string suggests, here are only 2^32
destroyers, so we have to ration them. You can skip this step,
but only at the expense of urbiting from a mere "submarine," a
self-generated 128-bit ship - like:
~bacbyl-dabdyr-marfel-palnep--tompex-nibnym-sidpet-ticfus
In theory a submarine can do anything a destroyer can. In
practice, if you have any interest in networking with other
ships - don't be surprised if they assume you're a bot.
Obviously, your ship is the root of your own personal namespace.
`~lacser-dovlet` is your "me function." Urbit, as a global
functional filesystem, is the union of everyone's me function.
(Urbit should not be confused with distributed storage systems,
like Freenet or IPFS. In Urbit all your data is on your own
computer, though it may be a virtual computer in the cloud.)
# The shape of urbitspace.
But wait. How, exactly, are these shorter names allocated?
As Samuel L. Jackson once put it: for money, bitch! But you're
an early adopter and we love you. So free ships for you, your
friends, and everyone.
Seriously - it will be a *long* time, if ever, before you have
any trouble getting a free Urbit destroyer, from us or anyone.
The main thing we want to make sure is that we're giving them
out mostly to unique humans. In the long run, this is quite hard
on the Internets. We're still small so no abuser gives a hoot.
And we (the evil Tlon Corporation, to be exact) are *not* the
only people who *can* give away destroyers, just the only people
doing it right now. Actually, some of the people who hold this
right are kind of our enemies - or at least, disgruntled former
employees (of the evil Tlon Corporation). They obviously can't
be trusted to collude in our evil schemes! So if you fear us,
you're probably safe with them.
Moreover, more importantly: once you own a destroyer (or more
realistically, once you sail it away from our harbor), it's
cryptographically yours and we (the evil Tlon Corporation) have
no practical way to steal it back, spy on you, etc. No, Urbit
isn't just another way of selling yourself into digital serfdom.
It's also important not to confuse Urbit with a cryptocurrency.
Urbitspace is digital land, not digital money. It is more like
the DNS than like Bitcoin. As in real estate, Urbit transactions
are unusual and have high frictional cost.
Making Urbit a bad currency is a feature, not a bug. Urbit is
designed to be neither particularly easy for governments to
control, nor particularly useful for criminals to abuse. Both
are difficult goals to hit, but they go well together. Any
system that attracts criminals will inherently also attract their
natural predator.
Also, as with land, the price of urbitspace is driven by direct
utility, not just collective speculation. And the only way land
is ever given away for free is as a homestead. Your free urbit
is for use, not for investment. Urbitspace will be worth
something if and only if it's useful to someone.
# More namespace design digressions.
Feel free to skip these if you really couldn't care less why.
Why are ship names synthetic strings, not user-chosen strings?
The simplest way to assign handles is with a central authority,
like Facebook. If you like being a digital serf, please, you
already know where to go. But even if you issue handles with a
proof-of-work scheme (like Namecoin), you still have a name rush.
And a decentralized system can have no way of negotiating with
real-world name owners whose existing claims conflict.
Also, there are concrete social benefits to a completely
impersonal address. An Urbit name is like a street name. It
doesn't mean anything and is not intended to. Even if it
accidentally means something, everyone ignores the accident -
when we get directions to a Bush Street or a Clinton Road, we
rarely even think of thinking of the politicians. Worse, handles
in real life tend to be just lame.
Why use short names at all, not just hashes? Why destroyers?
Why doesn't everyone just sail the Urbit seas in a submarine?
Two very important reasons. One, with only 2^32 names, an Urbit
name can be as memorable as a natural human name, or at least a
name in a foreign language. Two, when we have a limited supply
of names, network abusers don't find it so easy to disappear and
reappear under a new name.
(In fact we're very confident that it will be possible to keep
Urbit spam-free in practice. This would be harder if synthetic
short names were distributed as coins in a proof-of-work scheme.
Since Urbit treats namespace as property from day one, with no
"airdrop" to unproductive miners, there are no anonymous original
owners of Urbitspace. Everyone who owns a block has an incentive
to distributed it only to well-qualified individual users: as
with both IP addresses and actual real estate, Urbit blocks can
easily develop a collective bad (or good) reputation.)
It is of course possible to use Urbit anonymously, of course.
But only in one way: somewhere along the decentralization chain
from the (well-identified) developers to the user, someone has to
transfer urbitspace and be trusted to forget the recipient. It
is to be expected that, in order to provide this useful service,
the intermediary will demand some trust from the upstream users -
presumably, at least, evidence of humanity.
Point is: these are not the droids you're looking for. You are
not selling yourself into digital serfdom. You are escaping from
digital serfdom. You are not selling yourself into digital
serfdom. You are escaping from digital serfdom. Freedom is
slavery. Urbit is freedom. Now, please raise your right hand
and continue with the installation process.
# Install the SDK.
(Instructions here.)
If you got a destroyer, this will set up a local "yacht" on your
own computer. (Otherwise, just work out of your local
submarine.) Your destroyer lives in the cloud, where computing
is actually reliable. Leave it there. (You can sail it home, or
to some other harbor, any time you like.)
Suppose your destroyer is `~lacser-dovlet`, and your yacht is
`~sidpet-lacser-dovlet`. Urbit will make the directory
`~/urb/sidpet-lacser-dovlet/`, and symlink it, for convenience,
to `~/urb/lacser-dovlet/`.
This is a hotsync directory. When you edit, add or delete files
under the *pier* directory, `~/urb/sidpet-lacser-dovlet/`, it
syncs them automatically with your yacht, then syncs the yacht
with `~lacser-dovlet` in the sky. Sort of like Dropbox. Except
that (a) Dropbox is a hard disk and Urbit is a computer, and (b)
your Urbit computer is actually, in some sense, like, yours.
(Urbit is freedom. Repeat. Urbit is freedom.)
# Post and edit a webpage.
Start vere:
$ vere ~lacser-dovlet
Run vim on `~/urb/lacser-dovlet/main/pub/fab/fun/one.md`, and type:
This is a *fun experiment* in markdown.
Save the file. Then point a browser at
https://lacser-dovlet.urbit.org/pub/fab/fun/one
You'll see the rendered markdown. Then, change the file to read
This is a *different fun experiment* in markdown.
Save it. Then look at your browser. Did you have to hit reload?
*How exactly did that file get from your editor to your browser?*
# Urbit filesystem basics.
It's difficult when looking at this system superficially to
separate the web server from the namespace. Your namespace is
your namespace - the web server is just a legacy translation
layer from HTTP's rather non-functional namespace.
(In some senses it might make more sense to compute functionally
from Urbit's own command line. But your browser is a much better
renderer than anything in Urbit, and you're better at using it.)
We have seen two kinds of filenames go into Urbit: Unix filenames
and URLs. Obviously, neither of these is referentially
transparent. And neither is the same thing as an Urbit path, or
*beam*.
An Urbit beam has mostly the same syntax as a Unix path. The
beam always begins with the same three spans, `/ship/desk/case/`.
A ship is your computer, of course; a desk is a branch, project,
repository, etc; a case is a version. Versions are labels,
change numbers, or dates.
Besides the fact that revision control is part of the filesystem,
the only other weird thing about Urbit is that files and
directory are orthogonal. In Unix, you can't have one file
`/one/two/three` and another file `/one/two/three/four`, because
the `three` inode would have to be both a file and a directory.
Urbit uses different APIs for data and metadata requests, and
directories are implicit. The directory `/one/two/three` exists if
and only if there are files with that path as a prefix. There is
no `mkdir`, etc.
Without clunky stateful directories, our normal convention is to
use the last span of the path as the file type. So where Unix
has `/one/two/three/four.html`, Urbit has `/one/two/three/four/html`.
# Functional transformations.
In the test above, we started with a Unix filename, synced it
against an Urbit beam, and served it up as a URL. Ie, we went
from the Unix filename
~/urb/lacser-dovlet/main/pub/fab/fun/one.md
to the Urbit beam
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/md
to the URL
https://lacser-dovlet.urbit.org/pub/fab/fun/one
What happened internally? Without getting into too much
detail, the clean URL above was defaulted to
https://lacser-dovlet.urbit.org/pub/fab/fun/one.html
which then made Urbit try to serve the file
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/html
which did not exist, but we found your markdown source
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/md
and parsed it into the markdown tree
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/down
which was converted into the HTML tree
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/hymn
printed out into the HTML text file
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/html
and finally, wrapped up as the MIME object
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/one/mime
to ship back to you with a 200. Then, when you saved `one.md`
again, we did all this again and showed you the new document.
(We'll again delay on how *this* happened.)
# Functional construction.
This kind of format translation is a relatively straightforward
use of functional FS techniques. It's definitely handy but
doesn't really rise to the level of cool.
Here's something cool. Again with `vim`, edit
~/urb/lacser-dovlet/pub/fab/fun/two/down.hook
to be
:~ :- %par
:~ tex/"This is a "
emp/bent/~[tex/"fun experiment "]
tex/"in markdown."
== ==
Now, point your browser at
https://lacser-dovlet.urbit.org/pub/fab/fun/two
What did we do here? Instead of a static file containing
markdown text, we wrote a program in Hoon that generated
a markdown value. How did we find this? We went from
https://lacser-dovlet.urbit.org/pub/fab/fun/two.html
to the file you created,
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/two/down/hook
Then we followed a simple rule: you can make the file
`/one/two/three` by compiling and running the source file
`/one/two/three/hook`.
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/two/down
and then proceeded as above. You can test that this is really a
program, and still autoloads and stuff, by doing something funky
- like:
:~ :- %par
:~ tex/"This is a "
emp/bent/~[tex/"fun experiment "]
tex/"in markdown; 2 + 2 is {<(add 2 2)>}."
== ==
# Computing an actual function.
While these files are functions in a sort of abstract sense, they
are not actually functions in the normal sense of the word. A
function needs arguments, after all.
Since we're talking to our function through the Web, it makes
sense to try to send arguments as query strings. Also, though
we've had some fun generating markdown nouns (with totally
generic Hoon), it seems more appropriate since we're writing
webpages and stuff to generate HTML.
Let's edit
~/urb/lacser-dovlet/main/pub/fab/fun/three/hymn.hook
and put in
;html
;head ;title: Fun Experiment Three
==
;body
;p: This is an ;{i "HTML file."}
==
==
and view it at
https://lacser-dovlet.urbit.org/pub/fab/fun/three
We saw above how a `%hymn` turns into an `%html`, so we don't
need to repeat it. But again, this source file is best seen
as a program which produces an HTML tree. Sorta like the DOM.
(This syntax is also totally generic Hoon. In a sense - Hoon is
customized without apology for building XML nouns. In fact, it
has special cases for `<script`> and `<style>`, so it's even
slightly specific to HTML. This is not the most elegant language
design decision in the world, but it seemed preferable to making
people use a separate template language. Hoon is not the world's
best template language, but it's not just a template language.)
But what would be nicer would be -
=+ fib=|=(x=@ ?:((lth x 2) 1 (add $(x (dec x)) $(x (sub x 2)))))
;html
;head ;title: Fun Experiment Three
==
;body
;p ; This is an ;{i "HTML file"} which
; computes the Fibonacci number
; of 12: {<(fib 12)>}.
==
==
==
There's clearly a *function* here. But it's not *of* anything...
# An actual function, query string edition.
Okay, so we want an actual function. In
~/urb/lacser-dovlet/pub/fab/fun/four/hymn.hook
put
/= gas /$ fuel
::
=+ arg=(biff (~(get by qix.gas) %number) (slat %ud))
=+ fib=|=(x=@ ~+(?:((lth x 2) 1 (add $(x (dec x)) $(x (sub x 2))))))
::
;html
;head
;title: Fun Experiment Four
==
;body
;p: Welcome, {<cip.ced.gas>}!
;+ ?~ arg
;p: Usage: ?number=x
;p ; This is an ;{i "HTML file"} which
; computes the Fibonacci number
; of {<u.arg>}: {<(fib u.arg)>}.
==
==
==
Try it:
https://lacser-dovlet.urbit.org/pub/fab/fun/four?number=14
https://lacser-dovlet.urbit.org/pub/fab/fun/four?number=144
https://lacser-dovlet.urbit.org/pub/fab/fun/four
This page does appear to be a function of the query string - and
some other stuff. For one thing, Urbit is now tracking your
behavior. Or at least your IP address.
Actually, at the Web level, the page broadly speaking is a
function of the query string and the session state. The usual
Web cruft, basically.
However, the page is not computed at the chaotic Web level - it's
computed at the functional publishing level. Functional
publishing does not know the Web exists. We can reveal the true
and complete input to the page function with a sneaky trick.
In
~/urb/lacser-dovlet/pub/fab/fun/five/hymn.hook
put
/= ctx /$ |=([p=beam q=path] +<)
::
;html
;head
;title: Fun Experiment Five
==
;body
;p: This page was built at {<(tope p.ctx)>}.
;br;
;p: The remainder path was {<q.ctx>}.
==
==
Try it:
https://lacser-dovlet.urbit.org/pub/fab/fun/five
What you'll see is that `q.ctx`, which we used to be passing to
the Web path parser `fuel`, is an ugly (but URL-safe) string
containing data, encoded as a path component - let's say,
`web/DATA`. This string is actually inverted - properly
speaking, it's `DATA/web`.
When we want to compute a dynamic functional namespace which is
not just a trivial static file, or a translated file, we divide
the Urbit beam into two parts - *driver* and *remainder*.
The driver is the path to the function; the remainder is the
input to the function.
In this case, the Urbit webserver asked the filesystem for
/~lacser-dovlet/main/~2014.7.30/pub/fab/fun/five/DATA/web
with the content type `%html`. (We lied a little earlier - the
content type, or `mark`, is sent separately to the filesystem.)
To resolve the beam, we travel up it from the bottom, looking for
drivers of some sort. Urbit did not find any files in
`fun/five/DATA/web`. It also did not find any files in
`fun/five/DATA`.
Going up, it found `fun/five/hymn/hook`, which it took as a way of
making `fun/one` with the mark `%hymn`. Ie, an HTML document
tree, which we know how to turn into `%html`, so great.
Then, it gave the remainder of the beam, `/DATA/web` - but
inverted to `/web/DATA`, because we decode the remainder from
the end inward - as a resource to our page generator. And this
is how we could print your IP address and stuff...

View File

@ -0,0 +1,515 @@
# Ford changes for non-dummies.
You're used to .hoon files being parsed into a ++twig. This
stays true for vanes, hoon.hoon, and %batz apps; but for
everything else we get a ++hood.
It happens that every old .hoon file is also a ++hood. but there
are some extra runes at the top which do interesting cool things.
# Top-level file stuff.
The first, simplest thing that the new build system gives you:
without using any other features at all, we treat your file as a
list of nested twigs, ie `=~`, not a single twig. So instead of
=> |%
++ library
--
|%
++ program
--
you can write - using my new, official, stylized-tilde divider:
:: One line that describes this source file.
::::
:: /hoon/file/fab/pub
|%
++ library
--
::
::::
::
|%
++ program
--
The conventional way of doing `!:` has also changed. The trouble
is that `!:` is a twig and it cannot come above ++hood runes.
Actually, the right place to apply !: is per core, and it should
stand out visually because it indicates immature code:
!: |%
++ library
--
But you should always use at least one ++hood directive, namely
`/?`. This is tied to the version of Urbit as a whole, which in
practice means the version of `%zuse` - also defined as ++zuse.
Thus, `zuse` is 314 and `hoon` is 164.
The assumption, currently shaky, is that backward compatibility
in both ++zuse and ++hoon will never be broken. This is
basically what it means to do Kelvin versioning. With
/? 314
you state that this code should work iff `(lte zuse 314)`.
So this is what the start of a normal .hoon file looks like:
:: At least one line that describes this source file.
::
:::: /hoon/file/fab/pub
::
/? 314
/- html, hymn, down, tlon-radio-a
/+ tlon-radio, tlon-twitter
::
:::: ~tasfyn-partyv, ~tomsyt-balsen, ~talsur-todres
::
|%
++ code
!!
--
Note that hood runes by convention use *four* spaces, making
them visually distinct from twig runes - and preserving the
Rastersysteme.
Formal comments: the path is the `++spur` (path after /===/) of
the file, which is generally stored inverted. The list of ships
is the list of people who have changed the code, oldest first.
Anyone whose (new) .hoon files do not match this template is
eligible for a yellow card! I mean it!
# Mad hood runes.
Let's go through these runes one by one, creating a wacky test
program as we go.
There are three main sections of a hood: hooves, horns, hoops.
They must be in the file in that order. Let's talk about them
in the *opposite* order, though.
# Indirect hoops (simple).
Hoops are the actual code of your program. You can buy two kinds:
$% [%& p=twig] :: direct twig
[%| p=beam] :: resource location
== ::
The program produced by a hood is a stack of twigs which are
composed by `=~`, ie, connected with `=>`. You can specify these
twigs directly, as in the above, or you can load them from Urbit.
Let's build a toy program, `mad`, that uses mad hood skillz.
First, let's load it with an indirect hoop. In
~/urb/pub/src/mad/fib.hoon
we put
|%
++ fib |=(x=@ ~+(?:((lth x 2) 1 (add $(x (dec x)) $(x (sub x 2))))))
--
and in
~/urb/pub/fab/mad/one/hymn.hook
we put
:: Our first experiment with major hood runes.
::
:::: /hoon/one/mad/fab/pub
::
/= gas /$ fuel
// /===/pub/src/mad/fib
::
:::: ~tasfyn-partyv
::
=+ arg=(biff (~(get by qix.gas) %number) (slat %ud))
::
;html
;head
;title: Mad Experiment One
==
;body
;+ ?~ arg
;p: Usage: ?number=x
;p ; This is an ;{i "HTML file"} which
; computes the Fibonacci number
; of {<u.arg>}: {<(fib u.arg)>}.
==
==
==
# Indirect hoops (advanced).
Indirect hoops have a still more amazing power - they can make
cores out of directories themselves.
Let's `cp -r` `fab/mad/one` to `fab/mad/two` and put, in
~/urb/pub/src/mad/tools/fib.hoon
the single line
|=(x=@ ~+(?:((lth x 2) 1 (add $(x (dec x)) $(x (sub x 2))))))
Then, in
~/urb/pub/fab/mad/two/hymn.hook
we put:
:: Our second experiment with major hood runes.
::
:::: /hoon/two/mad/fab/pub
::
/= gas /$ fuel
// /===/pub/src/mad/tools
::
:::: ~tasfyn-partyv
::
=+ arg=(biff (~(get by qix.gas) %number) (slat %ud))
;html
;head
;title: Mad Experiment Two
==
;body
;+ ?~ arg
;p: Usage: ?number=x
;p ; This is an ;{i "HTML file"} which
; computes the Fibonacci number
; of {<u.arg>}: {<(fib u.arg)>}.
==
==
==
Note that our function name *does not exist anywhere in the
source code*, except where it is called. Rather, it's implicit
from the filename.
It would be superfluous to observe that this mechanism is
recursive - you can create arbitrarily deep hierarchies of cores
whose symbolic namespace follows the source code tree.
Don't mistake indirect hoops, however, for a library loading
mechanism. Hoops may be used to *build* libraries, but `/+`
is used to load them. Indirect hoops are for splitting up your
program into separate parts, not for sharing code between
programs.
# Horns.
Another thing we like to do is compile data into our application.
No, really - one of the stupidest things in many programming
environment is dynamically loading static resources. We really
try not to do that in Urbit. (At least, this week we try!)
Before her hoops, the programmer can load a set of referentially
transparent `horns`, or resources.
In
~/urb/pub/fab/mad/res/hello/hymn.hook
we put:
;p: Hello, world.
and in
~/urb/pub/fab/mad/three/hymn.hook
we put:
:: This third experiment wears the horns.
::
:::: /hoon/three/mad/fab/pub
::
/= hello /: /===/pub/fab/mad/res/hello /hymn/
:
:::: ~tasfyn-partyv
::
::
;html
;head
;title: Mad Experiment Three
==
;body
;+ hello
==
==
What did we do? We loaded a typed, dynamically built resource
from the global namespace onto our reef.
Note the runic syntax above. You might suspect that this is a
tall mode, and you might be right. It condenses:
/= hello /:/======/res/hello:/hymn/
Or even:
Note also how we avoid repeating shared path spans.
Also, unlike twig runes, hood runes can be replaced with a
semantic codename:
/dub hello /see /======/res/hello /hymn/
# More horns.
Cool as it is, this example barely scratches the awesome that is
Urbit resource loading. For instance, in
~/urb/pub/fab/mad/res/bible
put, in `1.html`, `2.html`, and `3.html` respectively,
<p>The earth was without form, and void.</p>
<p>Then Cain slew Abel.</p>
<p>I bring not peace, but a sword.</p>
then, in
~/urb/pub/fab/mad/four/hymn.hook
put
:: This fourth experiment is profoundly biblical.
::
:::: /hoon/four/mad/fab/pub
::
/= bible /: /======/res/bible
/; |= a=(list (pair ,@ manx))
(turn a |=([* b=manx] b))
/@
/hymn/
::
:::: ~tasfyn-partyv
::
!:
;html
;head
;title: Mad Experiment Four
==
;body
;* bible
==
==
What happened here? The `/@` rune loads a list of files named by
decimals in ascending order, with the protocol `/hymn/` - the
result of parsing HTML. This produces not a list of hymns
(actually just `++manx`, the XML node type), but a list of pairs
`[number manx]`, so we need to strip the numbers with `/;`.
We can also understand `/:` a little better. It doesn't mean
"load the horn at this beam" - it means "set the beam at which
horns within this one are loaded." There is also its cousin
`/,`, which just descends within the current beam. The beam
starts as the path to the hook file itself.
Nothing about this system restricts the tree to a fixed depth -
we could have not just `/hymn/` in our bible directory,
but another `/@` or other arbitrary resource directives.
`/@` has several cousins for different naming schemes. `/|`
reads a folder named by `@dr`, ie, relative date. `/&` reads
one named by `@da`, ie, absolute date. An excellent way to
organize random events named only by time is to nest `/|` within
`/&`, eg, a tree of days with that day's events in it.
You can also use `/%` to load a map of unparsed names. Finally,
`/*` handles multiple heterogeneous subtrees with different
horns. You can cast the product of `/*` to a map with `/^`,
or you can leave it as a tuple. You can also build a simple list
of horns with `/.`.
Finally, we've already seen `/$`, which calls a parsing function
(`++fuel` for web remainders) on the location and remainder of
the file. And `/~` lets you stick a simple twig directly in your
resource tree, for computing "resources" entirely by hand.
Here is the structure of horns:
++ horn :: resource tree
$% [%ape p=twig] :: /~ twig by hand
[%arg p=twig] :: /$ argument
[%day p=horn] :: /| list by @dr
[%dub p=term q=horn] :: /= apply face
[%fan p=(list horn)] :: /. list
[%for p=path q=horn] :: /, descend
[%hub p=horn] :: /@ list by @ud
[%man p=(map span horn)] :: /* hetero map
[%nap p=horn] :: /% homo map
[%now p=horn] :: /& list by @da
[%saw p=twig q=horn] :: /; operate on
[%see p=beam q=horn] :: /: relative to
[%sic p=tile q=horn] :: /^ cast
[%toy p=mark] :: /mark/ static
== ::
See `%ford` for the structure of horns, and `++fair` in ford for
the exact syntax and semantics.
# Hooves: structures.
At the top of your hood comes the proper way of sharing code
between programs: `/-` and `/+`, for structure and library
loading respectively.
Let's try `/-` by reusing one of our earlier examples - this
time, with an actual type cast. In
~/urb/pub/fab/mad/five/down.hook
put
/- down
^- down
:~ :- %par
:~ tex/"This is a "
emp/bent/~[tex/"fun experiment "]
tex/"in markdown; 2 + 2 is {<(add 2 2)>}."
== ==
In plain English, we loaded the `++down` structure and cast our
hand-rolled markdown to it. Where did we get this code, exactly?
It came from `/===/sur/down/gate/hook`:
/? 314
/- *markdown
down
The `*` tells us that we are dealing not with a single gate, but
a core containing multiple gates. `/===/sur/markdown/core/hook`:
|%
++ down (list barb) :: markdown structure
++ barb :: block elements
$% [%had p=@ud q=(list shin) r=(unit tape)] :: depth, contents, id
[%hem p=manx] :: html tag
[%hot ~] :: horizontal rule
[%lie p=down] :: list element
[%lit p=? q=down] :: list
[%par p=(list shin)] :: paragraph
[%pre p=wall] :: preformatted text
[%quo p=down] :: blockquote
== ::
++ shin :: span elements
$% [%cod p=tape] :: inline code
[%cut ~] :: break
[%emp p=?(%bent %bold %both) q=(list shin)] :: emphasis
[%ike p=(list shin)] :: strikethrough
[%lin p=(list shin) q=tape r=(unit tape)] :: link
[%tex p=tape] :: text
== ::
--
We see that markdown isn't a trivial single structure. With `*`
in a hoof we do two things. One, we load a core not a gate.
Two, we implicitly stick an
=+ markdown
on your reef, so that you can use `down` where otherwise you'd
have to say `down:markdown`. `down/gate/hook` uses this to
export the naked `down` gate into the structure space.
What actually happens to all the resources you load with `/-`?
They go into a single core above the libraries, horns and hoops.
Moreover, when libraries, horns and hoops contain `/-` runes of
their own, indicating structure dependencies, *all* these
dependencies are collected into the single structure core.
# Hooves: global names and conflicts.
When we use a single symbol, like `down`, we indicate that we're
looking for a structure in the same ship, desk, and case as our
own. In other words, we're loading a local structure which is
tightly coupled to the system loading it.
We can also use global structures. Here the syntax is
name/label/~ship
The desk in the global case is bound to `%main`, because it's
assumed that if you're publishing your code to the world it
belongs on your `%main`.
But this still becomes an arm named `structure` in your structure
core. What happens if we depend in two places on two hooves,
with the same name but different labels or ships?
For now, we are conservative - all hooves in a hood must match
exactly. Eventually we'll parse conventional labels (for Kelvin
and semantic versioning) and perform automagic upgrades.
# Hooves: libraries.
Libraries are loaded much the same way as structures, except with
`/+`. The syntax is the same, except that there is no equivalent
of `*`. Library dependencies are also collected across the
entire build, like structure dependencies.
Libraries are assumed to be cores, and they are stacked one after
another in dependency order - if a depends on b, a obviously must
come after b.
Let's try using a library. In
~/urb/lib/example/core.hook
put
|%
++ fib |=(x=@ ~+(?:((lth x 2) 1 (add $(x (dec x)) $(x (sub x 2))))))
--
Then, in
~/urb/pub/fab/mad/six/hymn.hook
we put:
:: Our sixth experiment with major hood runes.
::
:::: /hoon/six/mad/fab/pub
::
/+ example
/= gas /$ fuel
::
:::: ~tasfyn-partyv
::
=+ arg=(biff (~(get by qix.gas) %number) (slat %ud))
;html
;head
;title: Mad Experiment Two
==
;body
;+ ?~ arg
;p: Usage: ?number=x
;p ; This is an ;{i "HTML file"} which
; computes the Fibonacci number
; of {<u.arg>}: {<(fib u.arg)>}.
==
==
==

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 1
title: I - Intro To Hoon
---
#Intro to Hoon
> Like all men in Babylon, I have been proconsul; like all, a slave.
> - The Lottery in Babylon

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 2
title: II - Moar Hoon Types
---
#Types
> What good is a phone call if you're unable to speak?
> - The Matrix

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 3
title: III - Hoon Computes
---
#Hoon Computes
> I've only been in love with a beer bottle and a mirror.
> - Sid Vicious

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 4
title: IV - Gates
---
#Gates
> Anyone who thinks he's original is merely ignorant.
> - Nicolás Gómez Dávila

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 5
title: V - Tiles
---
#Tiles
> Not to get knowledge, but to save yourself from having
> ignorance foisted upon you.

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 6
title: VI - Type Inference
---
#Type Inference
> Ever, as before, does Madness remain a mysterious-terrific,
> altogether infernal boiling-up of the Nether Chaotic Deep,
@ -961,4 +955,4 @@ perfectly capable of inferring that when you weld `(list a)` and
change might confuse.
In short: generic polymorphism is cool but wacky. Leave it to
the experts, please!
the experts, please!

View File

@ -1,10 +1,5 @@
---
layout: tutpage
axis: doc-hoon
categories: tut
sort: 7
title: VII - Odds, Ends, Quirks
---
#Odds, Ends, Quirks
##A bestiary of cores##
@ -1214,4 +1209,4 @@ In freehand mode (F-Hoon), do whatever the heck you want. Note
that while uppercase is not permitted in a symbol, `-` is,
suggesting a generally Lisp-like state of gross hyphenated
disorder. F-mode is best used for top-layer software which
nothing else is based on; prototyping and casual coding; etc.
nothing else is based on; prototyping and casual coding; etc.

View File

@ -0,0 +1,29 @@
#Urbit
We got tired of system software from the 1970s. So we wrote our own. From scratch.
##Nock, a minimal virtual machine
[Nock](https://github.com/urbit/urbit/blob/master/urb/zod/spec/nock/5.txt) is a
homoiconic combinator algebra, not much fancier than SKI combinators. The spec
fits on a T-shirt and gzips to 340 bytes. We never extend Nock or call out to Unix from it.
##Hoon, a typed functional language
Hoon is a strict, typed, functional language that compiles itself to Nock.
As a functional systems language, Hoon is especially good at metaprogramming,
self-virtualization, hotpatching; marshalling and validating untyped data;
decoding and encoding binary message formats. Hoon is designed for event
programming, so there is no concurrency model.
##Arvo, a functional operating system
Arvo is an event-driven server OS built on the same event library as node.js
(libuv). Unlike node.js, Arvo is written in Hoon, isolated from Unix and a
persistent single-level store. Arvo is modular. Present modules provide a
network messaging protocol, a REPL and task manager, a revision-controlled
filesystem, a text console, and an HTTP server.

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-nock
categories: tut
sort: 1
title: I - Intro To Nock
---
#Intro to Nock
> What one fool can do, another can.
> - Ancient Simian proverb

View File

@ -1,10 +1,4 @@
---
layout: tutpage
axis: doc-nock
categories: tut
sort: 2
title: II - Nock Is Easy
---
#Nock is Easy
> You get used to it. I don't even see the code. All I see is blonde, brunette, redhead.
> - The Matrix

View File

@ -1,10 +1,5 @@
---
layout: tutpage
axis: doc-nock
categories: tut
sort: 3
title: III - Using Nock
---
#Using Nock
> But are you crazy enough?
> - Point Break

View File

@ -0,0 +1,463 @@
#Setup
> Tlön is surely a labyrinth, but it is a labyrinth devised
> by men, a labyrinth destined to be deciphered by men.
> - Tlön, Uqbar, Orbis Tertius
Urbit runs on Unix machines only. It depends on:
+ gmp
+ libsigsegv
+ openssl
+ libssl-dev (Linux only)
+ ncurses (Linux only)
Currently we support OSX, Linux (not all distributions have been
tested) and \*BSD. There are no instructions for BSD, because
only people with a serious clue run BSD. Intrepid ninjas may
attempt ports to other OSes. If you're not an intrepid ninja,
try a VM (eg, VirtualBox) or use one of our AMIs.
###Amazon AMIs
There are public AMIs at the following locations:
us-west (oregon) ami-6cf88d5c
us-west (n. california) ami-78d4ec3d
us-east (n. virginia) ami-cd819ba4
These use Debian Wheezy, so you should be able to
ssh -i /path/to/your/key/file.pem admin@123.123.123.123
to get in.
Since the codebase is changing frequently, run the following once you have ssh-ed in.
cd /urbit/
git pull origin master
make clean; make;
Then jump to "Run" below to get rolling.
###Docker
[Docker](http://docker.io) is a very convenient way to get in to an Urbit ship quickly. `~simmev-rabryd` maintains the docker approach on GitHub [here](https://github.com/yebyen/urbinit).
Follow the instructions on the GitHub page, then proceed to "Run" below.
###OS X###
1. Do you have XCode? Type `gcc`. If it says `no input files`, you have XCode.
Otherwise, install XCode: `https://developer.apple.com/xcode/`, with the
command line tools.
2. Install dependencies. Pick either one of Homebrew or Macports, but not both.
- Homebrew? Type `brew`. If it does something, you have Homebrew.
Otherwise, `ruby -e "$(curl -fsSL https://raw.github.com/Homebrew/homebrew/go/install)"`
will install it.
And follow up with `brew install git gmp libsigsegv openssl`
This will ask you for the root password, which ideally you know.
- Macports? Type `port`. If it does something, you have Macports.
Otherwise go [here](http://www.macports.org/install.php "here").
Then `sudo port install git gmp libsigsegv openssl`
Enter your root password at the prompt.
###Linux (Ubuntu or Debian)###
1. `sudo apt-get install libgmp3-dev libsigsegv-dev openssl libssl-dev libncurses5-dev git make exuberant-ctags`
###Linux (AWS)###
1. `sudo yum --enablerepo epel install gcc git gmp-devel openssl-devel ncurses-devel libsigsegv-devel ctags`
###Get the source###
Urbit uses git for its revision control. Eventually, Urbit will use itself, but for now,
you need git. If you've followed the above instructions correctly then typing `git` in your terminal should do something.
If that works, run:
git clone https://github.com/urbit/urbit.git
to download Urbit from its repository.
If for some reason you have moral qualms about using Git, you can also just download and unzip `https://github.com/urbit/urbit/archive/master.zip`. This won't provide any version control
###Set up your enviroment###
`cd` to the unpacked Urbit directory you just created:
cd urbit
If this works, `ls urb` should show:
urbit.pill zod/
Great! Now, let's do some dirty Unix stuff to set up your environment.
If you know what this is doing, feel free to do it right. Otherwise:
echo "export URBIT_HOME=`pwd`/urb" >>~/.bash_profile
source ~/.bash_profile
To make sure this worked,
echo $URBIT_HOME
should show `/urb` within the current directory.
If this didn't work, you'll have to do this the hard way. run `vi ~/.bash_profile` and fix it.
###Build###
`make`. Sometimes things are just easy.
<h3 id="run">Run</h3>
Run `bin/vere -c mypier`, where `mypier` is a directory that doesn't yet exist.
All your state (an append-only log and a memory checkpoint) will live in this
directory. Its name doesn't matter and is not visible internally.
A _pier_ is an Urbit virtual machine that hosts one or more Urbit identities,
or _ships_. When you run `vere -c`, it automatically creates a 128-bit ship,
or `submarine`. Your name (a hash of a randomly-generated public key) will
look like:
~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted
First you'll see a string of messages like:
vere: urbit home is /Users/cyarvin/Documents/src/u3/urb
loom: mapped 1024MB
time: ~2013.9.1..03.57.11..4935
ames: on localhost, UDP 63908.
generating 2048-bit RSA pair...
and then it'll pause a little, 'cause this is slow, and then
saving passcode in /Users/cyarvin/.urbit/~magsut-hopful.txt
(for real security, write it down and delete the file...)
and, then, if the network gods are happy, your submarine will start pulling
down Arvo files:
+ /~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted/main/1/bin/ticket/hoon
+ /~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted/main/1/bin/reset/hoon
+ /~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted/main/1/bin/ye/hoon
+ /~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted/main/1/bin/ls/hoon
and the like. You'll see a couple pages of this stuff. Don't worry too much
about the details right now. Finally, you'll get the Arvo shell prompt (which
is also a Hoon REPL):
~machec-binnev-dordeb-sogduc--dosmul-sarrum-faplec-nidted/try=>
###Register###
Next, you need to decide whether a mere submarine is enough for
you right now. This monicker is a mouthful. You can stick with
it (for now), but you're going to need a wider xterm.
Which might be fine! However, please note that just by sending a
simple email, you can get a much better ship - a `destroyer`,
with a nice short name like
~waclux-tomwyc
Just email `urbit@urbit.org`, with your submarine in the subject.
We'll send you destroyers - not one, but _two_. Yes, two! Tell
us something cool in the body, and we'll send you even more.
If you have a destroyer, you need to configure it. Otherwise,
just stretch that xterm wide and skip to section 1.2.
Your destroyers will arrive in the form of `[ship ticket]` pairs.
Let's say one of your ships is `~waclux-tomwyc` and its ticket is
~ribdyr-famtem-larrun-figtyd
(What are these strings, anyway? Just random unsigned integers,
rendered in Hoon's syllabic base, `@p`.)
A new life awaits you on the off-world colonies! To begin, just
type at the prompt:
:begin ~waclux-tomwyc
and follow the directions. When the script completes, hit return
and you'll be the `~waclux-tomwyc` you wanted to be.
##Play with Arvo##
If all went well, you now have a nice short prompt:
~waclux-tomwyc/try=>
If all did not go well (send us another email), or you're just
too impatient to wait for your destroyer, you have a big long
prompt. Which is fine, really, just ugly - and all these
exercises will still work.
###Example commands###
Let's try a few quick things to stretch your fingers. Type these
command lines and you should see the matching results:
~waclux-tomwyc/try=> "hello, world"
"hello, world"
~waclux-tomwyc/try=> (add 2 2)
4
~waclux-tomwyc/try=> :hello %world
"hello, world."
~waclux-tomwyc/try=> :cat /=main=/bin/hello/hoon
::
:: /=main=/bin/hello/hoon
::
|= *
|= [planet=@ta ~]
^- bowl
:_ ~ :_ ~
:- %%
!>("hello, {(trip planet)}.")
What did you just do?
One, you used Arvo as a Hoon REPL to print the constant `"hello,
world"`, which is a fancy way to write the Nock noun
[104 101 108 108 111 44 32 119 111 114 108 100 0]
Two, you called the Hoon `add` function to see that two plus two
is four. Math seems to work the same on the off-world colonies.
Three, you ran the Arvo application `:hello` with the argument
`%world`, which is just a fancy way to write the atom
`431.316.168.567` (or, for non-Germans, `431,316,168,567`). You
might recognize it better as `0x64.6c72.6f77` - the ASCII
characters in LSB first order.
(Is Urbit German? Sadly, no. But all our noun print formats are
URL-safe, which dot is and comma isn't.)
And you (4) used the Arvo application :cat to print the Hoon file
/=main=/bin/hello/hoon
which, supposing your current date is
~2013.9.1..04.38.31..f259
(ie, September 1, 2013 at 4:38:31 GMT/LS25 plus 0xf259/65536
seconds), is equivalent to the global path
/~waclux-tomwyc/main/~2013.8.23..04.38.31..f259/bin/hello/hoon
which anyone in Urbit can, see and even use - but we're getting
ahead of ourselves.
###Control characters###
In any case, what we've seen is that Arvo is a dangerous and
powerful operating system which if handled improperly can cause
serious injury or loss of life. We exaggerate. Slightly.
The first thing you need to know is how to control this tool.
Try your arrow keys - you'll see that Arvo has traditional Unix
history editing. Up and down, left and right work, as do the
simple emacs controls:
^A go to beginning of line
^B left arrow
^D delete next character
^E go to end of line
^F right arrow
^K kill to end of line
^L clear the screen
^R search through history
^U kill the whole line
^Y yank (restore from kill ring)
Don't expect any other emacs (or even readline - this is not readline, it's
internal to Arvo) commands to work.
There are also some special control keys specific to Arvo. It's
a good idea to learn these first so that you feel in, um,
control.
First, we'll quit out of an infinite loop with `^C`:
~waclux-tomwyc/try=> :infinite
When you hit return at the end of this line, Arvo will appear to
hang. Do not be alarmed! This is not a bug - it means that
we've started running our infinite loop before printing the next
console prompt. Simply hit `^C`, and you'll see
! intr
~waclux-tomwyc/try=> :infinite
(There may be some stacktrace stuff before the `! intr`, depending
on whether your kernel was compiled with debugging.)
Hit `^U` to delete the line and escape from infinity. Arvo is a
deterministic OS; you interrupted it while processing an event
that would never terminate. It returns to the state it was in
before you hit return - as if nothing had ever happened.
You're probably used to using nondeterministic, preemptive OSes,
in which the difference between a waiting task and an
executing event isn't apparent to the user. Since Arvo is not
preemptive, it has two very different states: waiting and
working.
When Arvo is working, `^C` cancels the event it's working on.
This event never happened. Don't worry, nothing bad will happen
to your computer.
When Arvo is waiting, use `^D` to end the current task, which is
the task that's currently prompting you. If there is a live
prompt and the cursor is not at the end, `^D` will delete the
current character - as in Unix.
Try this by running
~waclux-tomwyc/try=> :begin
Do you have a ship and a ticket? yes
Then hit `^D` and you'll be back to the command prompt (which,
unlike in Unix, is not a task itself, but part of the OS).
We don't always want to kill the prompting task. We often want
to switch between tasks, or between tasks and the command line.
Sort of like switching between windows, except in a command line.
We do this with `^X`. Try
~waclux-tomwyc/try=> :begin
Do you have a ship and a ticket? yes
But hit `^X` instead of `^D`. You'll get a prompt again. Use
it:
~waclux-tomwyc/try=> :begin
~waclux-tomwyc/try=> :hello %world
"hello, world."
~waclux-tomwyc/try=>
Hit `^X` again:
~waclux-tomwyc/try=> :begin
~waclux-tomwyc/try=> :hello %world
"hello, world."
Do you have a ship and a ticket? yes
And finally, hit `^C` to kill the task.
Lastly, Arvo is a single-level store. Since it's not the '70s
anymore and disk is cheap, everything you do is saved for ever.
(In fact, it's saved in two ways - as a memory image and an event
log - so you, or the government if they haz your filez, can
repeat every computation you've ever performed.)
If the current prompt is just the shell prompt, `^D` on an empty
line will log out - as in Unix:
~waclux-tomwyc/try=>
oxford:~/urbit; pwd
/Users/cyarvin/urbit
oxford:~/urbit; echo "hello, world"
hello, world
oxford:~/urbit;
Then you can restart and be right back where you were - just
run `vere` without `-c`:
oxford:~/urbit; bin/vere mypier
vere: urbit home is /Users/cyarvin/urb
loom: loaded 9MB
time: ~2013.9.1..17.23.05..0cc1
ames: on localhost, UDP 60342.
http: live on 8080
rest: checkpoint to event 383
rest: old 0v1c.gkr1o, new 0v10.m4gdu
---------------- playback complete----------------
waclux-tomwyc/try=>
Use your arrow keys and you'll see your history is still there.
Arvo is indestructible and can be shut down however you like
without losing data. Also, starting a new task while an old
one is still running will kill the old one safely.
But don't try to operate the same ship on two Unix hosts at the
same time. This will confuse everyone, including yourself.
###System administration###
Sometimes we make changes to Hoon or Arvo (we never make changes
to Nock) and you need to update your ship.
There are two steps to updating. You need to get the new files,
and you need to install them. To get them:
~waclux-tomwyc/try=> :update
: /~waclux-tomwyc/arvo/2/hoon/hoon
: /~waclux-tomwyc/arvo/2/dill/hoon
: /~waclux-tomwyc/arvo/2/batz/hoon
To install them (the simplest, slowest, most general way):
~waclux-tomwyc/try=> :reset
%reset-start
%reset-parsed
%reset-compiled
%hoon-load
[%tang /~waclux-tomwyc/arvo/~2013.11.26..20.29.15..090f/zuse ~tirnux-latwex]
[%vane %a /~waclux-tomwyc/arvo/~2013.11.26..20.29.15..090f/ames ~tolryn-watret]
[%vane %b /~waclux-tomwyc/arvo/~2013.11.26..20.29.15..090f/batz ~donfex-ladsem]
[%vane %c /~waclux-tomwyc/arvo/~2013.11.26..20.29.15..090f/clay ~picsug-mitref]
[%vane %d /~waclux-tomwyc/arvo/~2013.11.26..20.29.15..090f/dill ~dilpex-laptug]
[%vane %e /~waclux-tomwyc/arvo/~2013.11.26..20.29.15..090f/eyre ~forbur-disben]
All of your state, including running tasks, will be unchanged.
Sometimes the interpreter, called `vere` gets updated. In your urbit directory, back in Unixland, run:
git pull origin master
every so often to get the latest Urbit source code.
###Chat###
Okay, fine. You're a long way from being an Arvo ninja. But -
you're ready for the two most important uses of Urbit right now.
One, coding. Two, chatting.
To start coding, read the next chapter. To start chatting,
simply type
~waclux-tomwyc/try=> :chat
&
and type `?` for help.
[On to the documentation.](/doc/)

View File

@ -1,15 +0,0 @@
<div class="footer">
<div class="container">
<ul class="sidebar-nav">
<li class="sidebar-nav-item">
<a href="mailto:urbit@urbit.org">urbit@urbit.org</a>
</li>
<li class="sidebar-nav-item">
<a href="/community/articles/martian-computing/">544 Kelvin</a>
</li>
<li class="sidebar-nav-item">
<a href="/community/articles/continuity/">Continuity breach: ~2014.5.30</a>
</li>
</ul>
</div>
</div>

View File

@ -1,39 +0,0 @@
<head>
<link href="http://gmpg.org/xfn/11" rel="profile">
<meta http-equiv="content-type" content="text/html; charset=utf-8">
<!-- Enable responsiveness on mobile devices-->
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1">
<title>
{% if page.title == "Home" %}
{{ site.title }} &middot; {{ site.tagline }}
{% else %}
{{ page.title }} &middot; {{ site.title }}
{% endif %}
</title>
<!-- CSS -->
<link rel="stylesheet" href="/public/css/poole.css">
<link rel="stylesheet" href="/public/css/syntax.css">
<link rel="stylesheet" href="/public/css/main.css">
<!-- JS -->
<script type="text/javascript" src="//ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.min.js"></script>
<script type="text/javascript" src="/public/js/main.js"></script>
<script>
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
})(window,document,'script','//www.google-analytics.com/analytics.js','ga');
ga('create', 'UA-47054188-1', 'urbit.org');
ga('send', 'pageview');
</script>
<!-- Icons -->
<link rel="shortcut icon" href="/favicon.png">
<!-- RSS -->
<link rel="alternate" type="application/rss+xml" title="RSS" href="/atom.xml">
</head>

Binary file not shown.

Before

Width:  |  Height:  |  Size: 1.0 KiB

View File

@ -1,21 +0,0 @@
<div class="sidebar">
<div class="container sidebar-sticky">
<div class="sidebar-about">
<a href="/">
<h1 class="logo"><img src="/gen/main/pub/src/site/res/logo.png" class="logo"/></h1>
<h1 class="title">Urbit: a personal cloud computer</h1>
</a>
</div>
<ul class="sidebar-nav">
<li class="sidebar-nav-item">
<a href="/gen/main/pub/fab/site/documentation/" class="">Documentation</a>
</li>
<li class="sidebar-nav-item">
<a href="/community/" class="">Community</a>
</li>
<li class="sidebar-nav-item">
<a href="/setup/" class="">Setup</a>
</li>
</ul>
</div>
</div>