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         public override InputStream? run_async(Cancellable? cancellable) throws GLib.Error
73         {
74                 //var f = ensure_resource();
75
76                 var stream =  new GLib.MemoryInputStream.from_data (this.data.data,  GLib.free);
77
78                 return stream;
79         }
80         private async InputStream? run_impl(Cancellable? cancellable) throws GLib.Error
81         {
82             SourceFunc callback = run_impl.callback;
83             InputStream? ret = null;
84             new Thread<void*>("builder-fake-webserver", () => {
85                     // Actually do it
86                     try
87                     {
88                             ret = this.run_async(cancellable);
89                     }
90                     catch (Error e)
91                     {
92                             err = e;
93                     }
94
95                     // Schedule the callback in idle
96                     Idle.add((owned)callback);
97                     return null;
98             });
99
100             // Wait for it to finish, yield to caller
101             yield;
102
103             if (err != null)
104             {
105                     throw err;
106             }
107
108             // Return the input stream
109             return ret;
110         }
111         public void run((WebKit.URISchemeRequest request, Cancellable? cancellable) {
112             run_impl.begin(cancellable, (obj, res) => {
113                 InputStream? stream = null;
114
115                 try
116                 {
117                         stream = this.run_impl.end(res);
118                 } catch (Error e)  {
119                     request.finish_error(e);
120                 }
121                 if (stream == null)
122                 {
123                     stream = new MemoryInputStream();
124                 }
125                 request.finish(stream,
126                          this.size,
127                          this.content_type
128                  
129             
130                 });
131         }
132 }
133
134 public class FakeServer : Object
135 {
136         WebKit.WebView view;
137         
138         public FakeServer(WebKit.WebView wkview)
139         {
140                 this.view = wkview;
141                 
142                 
143                 // 
144                 
145                   
146         // Hook up signals.
147   
148         //this.view.resource_load_started.connect(on_resource_request_starting);
149         //this.view.navigation_policy_decision_requested.connect(on_navigation_policy_decision_requested);
150         //this.view.new_window_policy_decision_requested.connect(on_navigation_policy_decision_requested);
151           
152          //
153         var cx = this.view.get_context();
154         cx.register_uri_scheme("xhttp",  serve);
155         cx.set_cache_model (WebKit.CacheModel.DOCUMENT_VIEWER);
156         
157     }
158     
159     
160     public void serve(WebKit.URISchemeRequest request)
161     { 
162                 // request is URISchemeRequest
163                          
164                 print("REQ: %s\n",request.get_path());
165                 var cdata = FakeServerCache.factory(request.get_path());
166         
167                 if (cdata.size < 1 ) {
168                         print("Skip file missing = %s/gitlive%s\n", GLib.Environment.get_home_dir() , request.get_path());
169                         request.finish_error(new FakeServerError.FILE_DOES_NOT_EXIST ("My error msg"));
170                         return;
171                 }
172                 print("Send :%s, %s (%s/%d)", request.get_path(), 
173                       cdata.content_type, cdata.size.to_string(), cdata.data.length);
174            
175                 var stream = new GLib.MemoryInputStream.from_data (cdata.data.data,  GLib.free);
176                     
177                 // we could cache these memory streams... so no need to keep reading from disk...
178                 // then what happens if file get's updated - neet to check the data against the cache..
179                 
180                 
181                 
182                 request.finish (  stream, cdata.size  , cdata.content_type);
183                 //stream.close();
184         }
185
186    
187 }