src/Builder4/FakeServer.vala
[app.Builder.js] / src / Builder4 / FakeServer.vala
1 /**
2  * Originally this was supposed to intercept http calls and redirect them
3  * but that is not supported in webkit2 (without using the extension api)
4  * 
5  * so for now we have modified our server to serve use a base url of xhttp:
6  * 
7  * so all relative urls are based on that 
8  * 
9  * Idea is to serve the files from the file system, so no need to setup apache etc...
10  * This should work for the static content like css / javascript etc.. but 
11  * will cause issues with 'dynamic' xhr files (eg. the php stuff)
12  *
13  * the idea is nicked from geary.
14  * 
15  */
16 public errordomain FakeServerError {
17         FILE_DOES_NOT_EXIST
18 }
19
20 public class FakeServerCache : Object
21 {
22
23         public string data;
24         public string content_type;
25         public int64 size; 
26
27         public static Gee.HashMap<string,FakeServerCache> cache;
28         
29         public static FakeServerCache factory(string fname)
30         {
31             if (cache == null) {
32                 cache = new Gee.HashMap<string,FakeServerCache>();
33             }
34             if (cache.has_key(fname)) {
35                 return cache.get(fname);
36             }
37             var el = new  FakeServerCache(fname);
38  
39              
40             cache.set(fname, el);
41             return el;
42         }
43
44
45         public FakeServerCache( string fname) {
46                 var  file = File.new_for_path ( GLib.Environment.get_home_dir() + "/gitlive" + fname);
47                 if (!file.query_exists()) {
48                     this.data = "";
49                     this.content_type = "";
50                     this.size = 0;
51                     return;
52                 }
53                  var info = file.query_info(
54                                  "standard::*",
55                                 FileQueryInfoFlags.NONE
56                 );
57             this.content_type = info.get_content_type();
58             this.size = info.get_size();
59             string data;
60             size_t length;
61             GLib.FileUtils.get_contents(file.get_path(), out data, out length);
62
63             this.data = data;
64
65             print("FakeServerCache :%s, %s (%s/%d)\n", fname , 
66                   this.content_type, this.size.to_string(), this.data.length);
67             
68
69         }
70
71
72     
73         private async InputStream? run_impl(Cancellable? cancellable) throws GLib.Error
74         {
75             SourceFunc callback = run_impl.callback;
76             InputStream? ret = null;
77             new Thread<void*>("builder-fake-webserver", () => {
78                     // Actually do it
79                     try
80                     {
81                             ret = this.run_async(cancellable);
82                     }
83                     catch (Error e)
84                     {
85                             err = e;
86                     }
87
88                     // Schedule the callback in idle
89                     Idle.add((owned)callback);
90                     return null;
91             });
92
93             // Wait for it to finish, yield to caller
94             yield;
95
96             if (err != null)
97             {
98                     throw err;
99             }
100
101             // Return the input stream
102             return ret;
103         }
104         public void run((WebKit.URISchemeRequest request, Cancellable? cancellable) {
105             run_impl.begin(cancellable, (obj, res) => {
106                 InputStream? stream = null;
107
108                 try
109                 {
110                         stream = this.run_impl.end(res);
111                 } catch (Error e)  {
112                     request.finish_error(e);
113                 }
114                 if (stream == null)
115                 {
116                     stream = new MemoryInputStream();
117                 }
118                 request.finish(stream,
119                          this.size,
120                          this.content_type
121                  
122             
123                 });
124         }
125 }
126
127 public class FakeServer : Object
128 {
129         WebKit.WebView view;
130         
131         public FakeServer(WebKit.WebView wkview)
132         {
133                 this.view = wkview;
134                 
135                 
136                 // 
137                 
138                   
139         // Hook up signals.
140   
141         //this.view.resource_load_started.connect(on_resource_request_starting);
142         //this.view.navigation_policy_decision_requested.connect(on_navigation_policy_decision_requested);
143         //this.view.new_window_policy_decision_requested.connect(on_navigation_policy_decision_requested);
144           
145          //
146         var cx = this.view.get_context();
147         cx.register_uri_scheme("xhttp",  serve);
148         cx.set_cache_model (WebKit.CacheModel.DOCUMENT_VIEWER);
149         
150     }
151     
152     
153     public void serve(WebKit.URISchemeRequest request)
154     { 
155                 // request is URISchemeRequest
156                          
157                 print("REQ: %s\n",request.get_path());
158                 var cdata = FakeServerCache.factory(request.get_path());
159         
160                 if (cdata.size < 1 ) {
161                         print("Skip file missing = %s/gitlive%s\n", GLib.Environment.get_home_dir() , request.get_path());
162                         request.finish_error(new FakeServerError.FILE_DOES_NOT_EXIST ("My error msg"));
163                         return;
164                 }
165                 print("Send :%s, %s (%s/%d)", request.get_path(), 
166                       cdata.content_type, cdata.size.to_string(), cdata.data.length);
167            
168                 var stream = new GLib.MemoryInputStream.from_data (cdata.data.data,  GLib.free);
169                     
170                 // we could cache these memory streams... so no need to keep reading from disk...
171                 // then what happens if file get's updated - neet to check the data against the cache..
172                 
173                 
174                 
175                 request.finish (  stream, cdata.size  , cdata.content_type);
176                 //stream.close();
177         }
178
179    
180 }