| 
									
										
										
										
											2010-09-28 00:45:31 -04:00
										 |  |  | ! Copyright (C) 2008, 2010 Slava Pestov. | 
					
						
							| 
									
										
										
										
											2008-06-16 04:34:17 -04:00
										 |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | USING: namespaces make assocs sequences kernel classes splitting | 
					
						
							|  |  |  | words vocabs.loader accessors strings combinators arrays | 
					
						
							| 
									
										
										
										
											2009-02-06 11:54:13 -05:00
										 |  |  | continuations present fry urls http http.server xml.syntax xml.writer | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | http.server.redirection http.server.remapping ;
 | 
					
						
							| 
									
										
										
										
											2008-06-16 04:34:17 -04:00
										 |  |  | IN: furnace.utilities | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : word>string ( word -- string )
 | 
					
						
							| 
									
										
										
										
											2008-12-03 20:10:41 -05:00
										 |  |  |     [ vocabulary>> ] [ name>> ] bi ":" glue ;
 | 
					
						
							| 
									
										
										
										
											2008-06-16 04:34:17 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : words>strings ( seq -- seq' )
 | 
					
						
							|  |  |  |     [ word>string ] map ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: no-such-word name vocab ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : string>word ( string -- word )
 | 
					
						
							|  |  |  |     ":" split1 swap 2dup lookup dup
 | 
					
						
							|  |  |  |     [ 2nip ] [ drop no-such-word ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : strings>words ( seq -- seq' )
 | 
					
						
							|  |  |  |     [ string>word ] map ;
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : nested-responders ( -- seq )
 | 
					
						
							|  |  |  |     responder-nesting get values ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : each-responder ( quot -- )
 | 
					
						
							|  |  |  |    nested-responders swap each ; inline
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-28 00:45:31 -04:00
										 |  |  | ERROR: no-such-responder responder ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : base-path ( string -- seq )
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  |     dup responder-nesting get
 | 
					
						
							| 
									
										
										
										
											2011-10-24 20:00:09 -04:00
										 |  |  |     [ second class-of superclasses [ name>> = ] with any? ] with find nip
 | 
					
						
							| 
									
										
										
										
											2010-09-28 00:45:31 -04:00
										 |  |  |     [ first ] [ no-such-responder ] ?if ;
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : resolve-base-path ( string -- string' )
 | 
					
						
							|  |  |  |     "$" ?head [ | 
					
						
							|  |  |  |         [ | 
					
						
							|  |  |  |             "/" split1 [ base-path [  "/" % % ] each "/" % ] dip % | 
					
						
							|  |  |  |         ] "" make | 
					
						
							|  |  |  |     ] when ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : vocab-path ( vocab -- path )
 | 
					
						
							|  |  |  |     dup vocab-dir vocab-append-path ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : resolve-template-path ( pair -- path )
 | 
					
						
							|  |  |  |     [ | 
					
						
							|  |  |  |         first2 [ vocabulary>> vocab-path % ] [ "/" % % ] bi*
 | 
					
						
							|  |  |  |     ] "" make ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: modify-query ( query responder -- query' )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: object modify-query drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: modify-redirect-query ( query responder -- query' )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: object modify-redirect-query drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: adjust-url ( url -- url' )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: url adjust-url | 
					
						
							|  |  |  |     clone
 | 
					
						
							|  |  |  |         [ [ modify-query ] each-responder ] change-query | 
					
						
							|  |  |  |         [ resolve-base-path ] change-path | 
					
						
							|  |  |  |     relative-to-request ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: string adjust-url ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: adjust-redirect-url ( url -- url' )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: url adjust-redirect-url | 
					
						
							|  |  |  |     adjust-url | 
					
						
							|  |  |  |     [ [ modify-redirect-query ] each-responder ] change-query ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: string adjust-redirect-url ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: link-attr ( tag responder -- )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: object link-attr 2drop ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-08 21:32:11 -05:00
										 |  |  | GENERIC: modify-form ( responder -- xml/f )
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-08 22:17:59 -05:00
										 |  |  | M: object modify-form drop f ;
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-08 21:32:11 -05:00
										 |  |  | : form-modifications ( -- xml )
 | 
					
						
							| 
									
										
										
										
											2009-02-08 22:17:59 -05:00
										 |  |  |     [ [ modify-form [ , ] when* ] each-responder ] { } make ;
 | 
					
						
							| 
									
										
										
										
											2009-02-08 21:32:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : hidden-form-field ( value name -- xml )
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  |     over [ | 
					
						
							| 
									
										
										
										
											2009-02-05 15:34:55 -05:00
										 |  |  |         [XML <input type="hidden" value=<-> name=<->/> XML] | 
					
						
							| 
									
										
										
										
											2009-02-06 12:44:58 -05:00
										 |  |  |     ] [ drop ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-23 22:40:17 -05:00
										 |  |  | CONSTANT: nested-forms-key "__n" | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : referrer ( -- referrer/f )
 | 
					
						
							|  |  |  |     #! Typo is intentional, it's in the HTTP spec! | 
					
						
							|  |  |  |     "referer" request get header>> at
 | 
					
						
							| 
									
										
										
										
											2010-10-07 02:00:38 -04:00
										 |  |  |     dup [ >url ensure-port [ remap-port ] change-port ] when ;
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : user-agent ( -- user-agent )
 | 
					
						
							|  |  |  |     "user-agent" request get header>> at "" or ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : same-host? ( url -- ? )
 | 
					
						
							|  |  |  |     dup [ | 
					
						
							|  |  |  |         url get [ | 
					
						
							|  |  |  |             [ protocol>> ] | 
					
						
							| 
									
										
										
										
											2010-10-07 02:00:38 -04:00
										 |  |  |             [ host>> ] | 
					
						
							|  |  |  |             [ port>> remap-port ] | 
					
						
							|  |  |  |             tri 3array
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  |         ] bi@ =
 | 
					
						
							|  |  |  |     ] when ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : cookie-client-state ( key request -- value/f )
 | 
					
						
							|  |  |  |     swap get-cookie dup [ value>> ] when ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : post-client-state ( key request -- value/f )
 | 
					
						
							|  |  |  |     request-params at ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : client-state ( key -- value/f )
 | 
					
						
							|  |  |  |     request get dup method>> { | 
					
						
							|  |  |  |         { "GET" [ cookie-client-state ] } | 
					
						
							|  |  |  |         { "HEAD" [ cookie-client-state ] } | 
					
						
							|  |  |  |         { "POST" [ post-client-state ] } | 
					
						
							|  |  |  |     } case ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SYMBOL: exit-continuation | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-23 22:40:17 -05:00
										 |  |  | : exit-with ( value -- * )
 | 
					
						
							| 
									
										
										
										
											2008-11-24 21:26:11 -05:00
										 |  |  |     exit-continuation get continue-with ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : with-exit-continuation ( quot -- value )
 | 
					
						
							| 
									
										
										
										
											2009-03-15 19:19:29 -04:00
										 |  |  |     '[ exit-continuation set @ ] callcc1 exit-continuation off ; inline
 |