IRCaBot 2.1.0
GPLv3 © acetone, 2021-2022
#saltr
/2023/10/21
y2kboy23 dr|z3d snark refresh is broken. Seeing "Uncaught TypeError: filterbar is null" in the dev console.
y2kboy23 well, once I enabled it, it started refreshing
T3s|4 y2kboy23: do you mean once I enabled the 'Torrent filter bar'?
dr|z3d y2kboy23: thanks, will fix that.
dr|z3d on the plus side, no pun intended, you got exposed to a feature you obviously weren't using :)
dr|z3d T3s|4: have a look at snark's configs, you'll see what y2kboy23 means.
y2kboy23 Sorry. With the filter bar off, the refresh is broken.
zzz dr|z3d, I think you're in desperate need of concurrency help?
dr|z3d if you're offering, zzz, I'll take any help I can get, thanks. it's what I'm working on right now.
zzz yup
dr|z3d synchronized isn't cutting it. I was going to move the removal of stale entries inside the writer, but haven't got there yet.
zzz first of all, I recommend the concurrency section of Thinking In Java
zzz but here's a tl;dr
dr|z3d concurrentHashMap I tried, no dice there either.
zzz when using non-thread-safe data structures such as HashMap, ArrayList, etc., there's only 3 options to avoid ConcurrentModificationExceptions and weird bugs
zzz 1) ensure ALL accesses (get(), put(), remove(), clear(), etc) are from a single thread
zzz 2) synchronize around ALL accesses with the same lock
zzz 3) switch to a thread-safe-datastructure such as ConcurrentHashMap
zzz there's an additional requirement when removing an element inside a loop
zzz let me look at your takes, just a sec
dr|z3d you've summarized nicely what I've discovered. I think I was/am probably omitting the file lock element with synchronized which is why it's not working as intended.
zzz no. you need to synch at _every_single_place_ you touch rdnscache
zzz and yes, to remove inside a loop, you must use iter.remove(), and you must never add within a loop if you're looping over that thing
dr|z3d yeah, got that. my local copy has that implemented.
dr|z3d the synchronized everywhere part.
zzz ConcurrentHashMap should also work, subject to the loop requirements above
zzz that's it, have fun
dr|z3d thanks, appreciate the insights.
dr|z3d this is the current iteration of the relevant methods, fwiw. cake.i2p/view/nzTmK9ETeA_N28jAPcD81cTS1ZRg8dr2c2UJh2B0W_OpEmw4X4aU/nzTmK9ETeA.txt
dr|z3d I noticed you got the rough end of Opicaak earlier. Ugly :|
zzz you can see why java 5's ConcurrentHashMap was a godsend
dr|z3d Yeah, I obviously didn't do it right when I tried a ConcurrentHashMap, it got rid of the errors but also didn't bother to read in the filecache to memory.
dr|z3d I'm going to add a couple more synchronized to that code block I just caked and see what gives.
dr|z3d the error took a while to manifest because stale entries in the file cache are only removed when the # entries exceeds the max configured size, 32K in most cases.
zzz you also need to synch readRDNSCacheFromFile, and lots of spots that are not in that pastebin
dr|z3d yup, done that, testing code.
dr|z3d what's your appraisal of the general idea? nuts? or? :)
zzz mildly nuts
dr|z3d It makes for a) a much more informative console when looking at tables and the netdb.
dr|z3d and b) with a file cache, zero delay once the cache is populated and minimal lookups.
zzz one other subtlety: public synchronized static foo() is a DIFFERENT lock than public synchronized bar()
dr|z3d so all methods being synchronized on the same file need to be either static, or not.
zzz yup
dr|z3d ok, good to know, thanks. I'm all static for the file cache methods.
zzz the former is the same as synchronized(BlahBlah.class) { ... } and the latter is the same as synchronized(this) { ... }
zzz the best practice would be to do neither, but to synch on the object in question, i.e. synchronized(rdnsCache) { ... } everywhere
zzz but your way will work
dr|z3d current revision, will take a few minutes for errors to generate, or not: cake.i2p/view/DlyObgeFeT_j54qTDyqRWx3EtCyrr8RLhpXFAPckk_jZUw0X25KO/DlyObgeFeT.txt
dr|z3d a fully populated 32K entry file cache is ~1MB, which isn't too bad.
dr|z3d as I thought, still the same error.
dr|z3d this method:
dr|z3d private static synchronized void cleanupRDNSCache() {
dr|z3d List<String> keysToRemove = new ArrayList<>();
dr|z3d Iterator<Map.Entry<String, CacheEntry>> it = rdnsCache.entrySet().iterator();
dr|z3d while (it.hasNext()) {
dr|z3d Map.Entry<String, CacheEntry> entry = it.next();
dr|z3d if (rdnsCache.size() > MAX_RDNS_CACHE_SIZE) {
dr|z3d keysToRemove.add(entry.getKey());
dr|z3d it.remove();
dr|z3d rdnsCache.keySet().removeAll(keysToRemove);
dr|z3d really does not like Map.Entry<String, CacheEntry> entry = it.next()
zzz well, for one, you either need it.remove() or the whole keysToRemove thing, dont do both
zzz it.remove() is a lot more efficient
dr|z3d ok, I'll try that, thanks.
zzz but that doesn't explain why it's not working
T3s|4 dr|z3d: yep, on the snarkified 'plus' side, seems to have been working / is working as expected :)
zzz did you sync the methods below cleanupRDNSCache() that you did not include in the paste?
dr|z3d thanks for reporting back, T3s|4. y2kboy23's issue was easily fixed, I put something in last minute without testing the badge refresh properly.
T3s|4 great
dr|z3d zzz: this one, yes, but I have a feeling there's another one that needs the syncrhonized treatment:
dr|z3d public static synchronized int countRdnsCacheEntries() {
dr|z3d return rdnsCache.size();
dr|z3d like probably this one: public String getCanonicalHostName(String ipAddress) {
zzz bingo
dr|z3d although not sure about that, and it's not a static method.
dr|z3d let's see if it moans if it's converted to static :)
zzz you'll have to take out the logging
dr|z3d yeah, I was using system.err where the logging wasn't working.
dr|z3d will remove it once the whole thing is working properly, don't want to be polluting wrapper logs.
zzz FYI, java overhead for a Map entry is about 120 bytes, and the way you did it is even more, so your size estimate is at least 4X low
dr|z3d talking about the on-disk cache.
dr|z3d I figured the in-mem cache would be larger, 4x filecache, not bad. 4MB when it's full on a >=512M router, acceptable.
zzz why in the world is the map IP->CacheEntry(IP, hostname) rather than just IP->hostname ??? Why store the key inside the value?
zzz that's adding about another 56 bytes per entry
dr|z3d that's a good question. :)
dr|z3d so I had to implement a lock object to bridge the static and non-static methods.
dr|z3d testing code, let's see if it works.
dr|z3d so it looks like this might be working, thanks for the help, zzz. I'll have a crack at the cache entry storage next. cake.i2p/view/1np26pHE10_RGppfzDvzYLxRIyE76Z6QMKwagrzv3_CPRrXEIH6b/1np26pHE10.txt
zzz ip->hostname would disallow null hostnames; not sure if you want to store those as a kind of negative lookup cache mechanism
zzz you'd have to switch to a separate negative lookup cache, or use something else like "" to indicate you already looked it up
zzz ja, though as I said if you're doing that, do the best practice of syncing on rdnsCache itself (and make it final), no need for a separate lock
zzz but your way works fine
dr|z3d that latest cake should address the ip/hostname issue in the mem cache.
dr|z3d the lock stuff is working a.ok, no more concurrent mod errors in console.
dr|z3d noted re syncing on the cache, thanks.
zzz no, my suggestion is Map<String, String> rdnsCache, making it a map of IP -> hostname, and nuking the CacheEntry class completely
zzz this would vastly simplify it but would be a big refactor
zzz there should never be a need to store the key inside the value, that's what a Map does for you, it associates a key to a value
dr|z3d yeah, right, so I've removed that, the ip is now the key.
zzz ok, you're fast, but it's not in the last cake above
dr|z3d oh, then I must be doing it wrong. I thought: rdnsCache.put(cacheEntry.getIpAddress(), cacheEntry); would address it.
dr|z3d more work to do, thanks. will ping you again when I'm sure I've fixed it, though possibly not with a total refactor just yet.
zzz I'm saying you don't need class CacheEntry at all. rdnsCache.put(ip, hostname); hostname = rdnsCache.get(ip)
zzz but yes it's a big change to something you just got working, wouldn't blame you if you didn't, it's just not great code
zzz I'm saying you don't need class CacheEntry at all. rdnsCache.put(ip, hostname); hostname = rdnsCache.get(ip)
zzz but yes it's a big change to something you just got working, wouldn't blame you if you didn't, it's just not great code
zzz the map "keeps" the key and value "together". You don't need to put them both in a CacheEntry class to do that
dr|z3d hey, it works, we're not all java sexperts like you :)
dr|z3d but thanks for the feedback, will look at a refactor maybe in the next release cycle.
Irc2PGuest60863 it might be useful to have a CacheEntry if you want to keep timestamp with the hostname
Irc2PGuest60863 (which is usually how caches work, remove the oldest or lru)
Irc2PGuest60863 (also pretty sure jdk has a built in lru cache)
dr|z3d I don't want a timestamp with the hostname, just makes the cache bigger. We're storing the cache entries oldest first, so they get removed when the cache is full.
Irc2PGuest60863 Blinded message
Irc2PGuest60863 what makes you think you're storing the oldest first?
Irc2PGuest60863 though admittedly I just glanced at the code
zzz right
zzz a HashMap don't do that
Irc2PGuest60863 I could've sworn java has a built in class that does this
zzz LinkedHashMap
Irc2PGuest60863 yeah LinkedHashMap gets you 90% of the way there I think
dr|z3d patches welcome, as is the common refrain :)
Irc2PGuest60863 what version of java do you use anyways? is the router still restricted to java8?
zzz our LHMCache class gets you the other 10%
dr|z3d Java8 is what we're currently requiring.
dr|z3d I feel like you want to pull this code into canon, zzz :)
Irc2PGuest60863 how do you retype the code so fast hehe
zzz we have a no nuts policy in canon
dr|z3d you need to re-evaluate the code. this is lovely, ripe fruit :)
Irc2PGuest60863 dr|z3d: I worry about that writeRNDSCacheToFile method which calls isDuplicateEntry on every line which re-reads the file?
Irc2PGuest60863 but the code seems fine. If I was really reviewing the code I might say it's not... OO enough. But some people don't like my OO style. Also obtaining a lock to write stuff to disk is rarely necessary
Irc2PGuest60863 in java 99% of the time you're better off making a copy of a data structure and then writing that then synchronizing
Irc2PGuest60863 then again the cost of synchronization is so low these days eh
Irc2PGuest60863 I'll also always wonder about the use of statics within the router codebase
Irc2PGuest60863 none of that code needs to be static afaict
dr|z3d fire up git, pull the code, see if you can improve it.
Irc2PGuest60863 hehe he said fire up git
Irc2PGuest60863 git's an exe
dr|z3d I'll push the latest revision once I've determined it's not hosing stuff.
Irc2PGuest60863 dr|z3d: I'll improve it if you can tell me why I haven't been able to load postman all day heh
dr|z3d what version are you running?
dr|z3d of i2p+?
Irc2PGuest60863 I'm actually genuinely curious what's going on. Before I thought it was a matter of router integration but it's been 27 hours
dr|z3d 58+ perchance?
Irc2PGuest60863 dr|z3d: yeah I'm running the latest version of i2p+ as usual. everything else seems to be working fine... except postman
dr|z3d is that 58+ or something more recent?
dr|z3d because 65+ is current right now.
Irc2PGuest60863 really I guess I could upgrade again. you think it might help
dr|z3d regarding postman, I don't know. I'd wait a few if you are going to upgrade, I'll push a new update with the RDNS changes once I'm confident they're working as intended.
dr|z3d I'm reverting the linkedHashMap and key/value changes for now, not working as intended, file cache is getting overwritten. Need to look at that again later.
dr|z3d ok, -66+ uploaded.
zzz postman working for me on canon trunk
Irc2PGuest60863 I will note that (1) you don't want to copy the tmp file and then delete it. Instead you want to use an atomic file move. That code as is is buggy and will break because copy isn't atomic.
Irc2PGuest60863 (2) you don't need t lock the map to write it to disk, just create a copy. Map<String,String> copyToWriteToDisk = new HashMap<>(ipToHostname) does the trick
Irc2PGuest60863 (3) I still don't understand what isDuplicateentry is doing in that code. it's pretty strange.
Irc2PGuest60863 lastly I'm never a fan of developers creating their own adhoc file formats, hehe. just use xml. But that dear friends is a story for another time
dr|z3d ip,hostname
dr|z3d it's pretty simple, keeps the file as small as possible, and is human readable.
Irc2PGuest60863 yeah probably, in this case, I'm just never a fan of people inventing file formats
dr|z3d it's .txt, it's not a new format.
Irc2PGuest60863 there's strange edge case you haven't thought of that would be handled by an existing file format
Irc2PGuest60863 even using a Java Properties Writer might be... safer
Irc2PGuest60863 also I might keep the Cache Entry class. The reason is you probably want to remember when you did the resolution and got the hostname. During initialization when you read in the file I would ignore any cache entries older than 30 days
dr|z3d as for isDuplicateEntry, it's meant to check for dupes.. the whole point of a simple format is to allow you to manually edit it, merge files if required.
Irc2PGuest60863 stuff like that happens when somebody installs i2p, stops running it, then comes back 2 years later and restarts the router and now the info in your cache might be wrong
Irc2PGuest60863 dr|z3d: why do you need to check for dupes in a map?
dr|z3d there's sufficient churn to not need to care about the timestamp.
dr|z3d you don't, you need to check for dupes in the file.
Irc2PGuest60863 dr|z3d: but if the only thing you ever write to the file is a map how could dupes get in the file?
dr|z3d I just told you.
dr|z3d you might manually merge files.
Irc2PGuest60863 hehe "sufficient churn to not need to care about the timestamp" == famous last words
Irc2PGuest60863 dr|z3d: when you read the file into a map in memory dupes will be eliminated
Irc2PGuest60863 when you write the file to disk there can be no dupes
Irc2PGuest60863 so all that code has to do is make sure to read the file and then write the file and use the ATOMIC_MOVE option not the copy operation
Irc2PGuest60863 and yes you've invented a new file format. just understand that's a risky thing to do. most file formats in i2p for example are accompanied by a formal spec. but yeah whatever
dr|z3d open file, see format. :)
dr|z3d which is the point. human readable. I don't want xml or anything else which obfuscates the information.
Irc2PGuest60863 xml is much more human readable than a custom format
dr|z3d actually, no. not in this case.
Irc2PGuest60863 but I digress, my main point was that you shouldn't need to check for duplicates
Irc2PGuest60863 and a timestamp isn't a crazy idea
dr|z3d ok, thanks, noted. I'll put that theory to the test later.
dr|z3d I initially had timestamps, just found them to be unnecessary, first in, first out is how the cache _should_ function.
Irc2PGuest60863 I'd also suggest that storing cached data in a config directory is strange. It's at best tmp data. the sort of thing you dont' really want to keep if you for example switch router versions. not sure the router policy on this but it's usually a good idea to distinguish between configuration data and runtime/"workspace" data and "tmp" data
dr|z3d no, it's definitely meant to be in .i2p/
Irc2PGuest60863 dr|z3d: timestamps aren't for eviction. they're a form of sanity checking. if you read in cache data which is, say, 2 years old, then it should all be thrown away
Irc2PGuest60863 you're assuming ppl are regularly restarting their routers... which is an assumption that won't always hold
dr|z3d first in, first out. like I said, there's enough churn to remove stale entries soon enough. the hostname data isn't critical.
dr|z3d no assumption relating to router uptime.
dr|z3d We just don't want to store cache data in a volatile location, it's in .i2p because it needs to be persistent.
zzz 'dupes impossible in a map' is not a theory
zzz 'why put the key on the value side of a key/value store' is neither a Java question nor a programming question
zzz I doubt the file is in ~/.i2p but if it is it's not on purpose
zzz the file format objection is silly but if you claim it's a csv file you might win the argument
zzz we have his ATOMIC_MOVE in our utils as FileUtil.rename()
zzz "xml is much more human-readable than csv" is an insane take
dr|z3d did I miss part of this conversation. not sure.
dr|z3d happy with the file format, that's not going to change.
zzz no, just comments on the last half of the rdns discussion where ircguestxxx got involved
dr|z3d location for the file is ~/.i2p/ in purpose.
dr|z3d *on purpose
zzz do me a favor and look if it's there
zzz on your disk
dr|z3d it is.
dr|z3d always has been. by design.
zzz not ~/i2p/ ?
zzz ah never mind, I see where you did it
zzz missed it because it's not how I would have done it
zzz see, told you it wouldn't be long before I was wrong
zzz ofc ~/.i2p is the right place, as he said he doesn't know what the 'router policy' is
dr|z3d you're right about xml, you'd have to be smoking crack to suggest that was more human readable than what I've implemented :)
zzz no, not just more -- "much more"
zzz you also presumably yanked that code from one of the 20 times I've done it, so I'll take your side on that one
dr|z3d right. and /tmp/ is definitely the wrong place.
dr|z3d I forget, but probably :)
dr|z3d the main thing right now is that it works and doesn't cause console errors. as you've pointed out, it could be made much more efficient and I'll have another pass at it post release, probably.
dr|z3d it's the difference between waiting 30s or more for one of the pages containing large peer tables to load and having the page load almost instantly. so I consider it a win, all things considered.
dr|z3d oh, and if it's not obvious, ircguestxxx is mesh.
dr|z3d who still hasn't reported back on the update notifications he specifically asked for.