-
Notifications
You must be signed in to change notification settings - Fork 46
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Cache map data in Kothic #37
Comments
Bobcat cache isn't suitable here, because we need to cache rendered tiles for short period to increase map dragging performance. |
this was posted to my e-mail today by unxed: В общем, как и собирался, попробовал я к котику приделать кэширование отрендеренных тайлов. Первое, что пришло в голову заюзать - LocalStorage. Начинаю понимать, что дело не только в доступе в LocalStorage. Если со временем починят скорость drawImage, можно попробовать кэшировать через IndexedDB Так что, похоже, идею с кэшированием есть смысл пока отложить до лучших времён, PS: в кэш сохранял в onRenderComplete, из кэша читал в drawTile. |
Here is my expierence in implementing some kind of caching in kothic-js (it is draft translation of russian comment above with some additions and fixes). As we cannot directly access browser's cache from javascript, we have to use some other mechanisms to store cached tiles on the client side.
Each of them has it's advantages and disadvantages. First of all, I think we should not use browser- or vendor-specific technologies so we should exclude Gears API from that list (not to mention the fact that Gears are no longer supported). Now we should exclude technologies with outdated/deprecated specs. Goodbye, WebSQL API (I personally will miss you...). So, now we have:
My first try was to use HTML5 LocalStorage. It was relatively easy to implement, but all in all it was not dramaticly faster. The first thing is that the size of LocalStorage is limited at 5 Mb in most browsers, and in Chrome, for example, you can't even change that limit. Okay, we can set 500 Mb in Firefox through about:config page. The second thing is that LocalStorage access in Chrome starts to slow down seriously after about ~500Kb of space used. I know no serious reson for this and consider it to be Chrome's bug. At that point I came to the conclusion that the problem is not only in LocalStorage speed, so I tried to make a simple tile cache in memory to see how fast it would be. As it was not much faster than LocalStorage-based cache, I started profiling the caching functions and found that the third thing that slows down cache implementation is drawImage function of canvas object (I used toDataURL to get tile's content and drawImage to place it on canvas from cache). Really, tiles are read from memory and rendered at about 0.7 second per tile! Yes, base64-decoding of data urls also takes some time, but the major time waste is inside drawImage call. Yes, it is somethat faster than to render all tiles as it is done currently. But it should be at least as fast as showing pre-rendered .png tiles (and maybe even faster as tiles are stored localy). Firefox has a bug ticket describing slow drawImage implementation. I don't know if it is releated to my problem or not: So if we will see (much) faster drawImage implementation in browsers in near future, we can try following:
PS: Most expirements was done in kothic-leaflet.js; I cached rendered tiles in onRenderComplete and read them from cache in drawTile. |
Here is sample implementation of caching using FileSystem API. Works (somehow) in Chrome. Not supported in Firefox. Not tested in other browsers/engines. tileprxy.php is simple json-tile-proxy script that caches json tiles on server side to prevent high load on osmosnimki.ru. It accepts x, y and z (zoom) as GET parametres and returns corresponding json tile. Remember, you should use some kind of proxy serverside script (or store json tiles on your own server) to follow browser's security policy. You cannot just replace var url ... string to "url = 'http://osmosnimki.ru/vtile/' + key + '.js';" - it will break everything. As you will see, Chrome (at least Chrome 14) leaks too much memory while caching is enabled so it will definitely crash sooner or later. Remember, this is proof-of-concept implementation, it should have bugs et cetera. |
Fixed stupid bug in FileSystem API cache implementation. Now _canvases array does not grow infinitely, so memory usage is seriously decreased. But strange Chrome crashes still happens from time to time - needs further investigation. |
Bug report related to Chrome's crashes after several FileReader calls: http://code.google.com/p/chromium/issues/detail?id=101668 |
Implemented caching tiles as binary PNG files, not as text files with base64-encoded PNG data as it had been done previously. https://gist.github.com/1319325 Demo with cache enabled: http://2g0.ru/map/kothic |
There is another way to gain some more speed. The following example eliminates the need to call drawImage to show images from cache by adding additional layer that shows cached png tiles directly from local filesystem. You should patch leafleat.js, replace https://gist.github.com/1320056 Demo as usual at http://2g0.ru/map/kothic |
Now, when tile moved out of visible area, the tile being completely removed from memory. We need to cache preprocessed data to reduce tile loading next time. Bobcat renderer already has cache implementation, so it can be easily copied.
The text was updated successfully, but these errors were encountered: