Monday, January 31, 2022
Monday, January 24, 2022
Saturday, January 22, 2022
Tuesday, January 18, 2022
Project 21 - Code Doodle - Introduction
As I've mentioned in my last few updates I'm starting to get a little bit more done on some of my *other* projects. Despite my best efforts though, some of those projects are big, and hard to finish and they need me to learn things, even when I think I've kept them small.
The solution to that is *clearly* to start doing something else as well. Because spreading myself thinly has really been a key to my success.
Oh.
Right.
No, it's definitely the other thing where I'm not great at finishing things and it's easier to play with the new shiny thing than it is to get shit done.
That being said, I am starting a new project and I have a bunch of ideas whey I think it's a good idea.
Ok Fine. I'll bite, what are you doing?
You Mentioned You Had Ideas as to Why This Wasn't a Bad Idea
So What Are You Doing?
- List Difference - takes a look at two lists and lets you see how similar they are. It's intended to help statistically mock sports casters. This one's pretty finished.
- Food-o-rac-o-cycle - named for the food machine on the Jetsons (I think). This will tell me what to make for breakfast. This one's off to a good start.
Monday, January 17, 2022
Tuesday, January 11, 2022
Blog: What I learned, forgot and learned again about Java Preferences
I'm not an expert in Java, but I have been using and teaching it for a really long time. One thing I've always struggled with is the right way to manage settings and preferences, especially stacking different levels of settings (defaults, setup time and run time). The various systems I worked on in grad school were massively over complicated (to the extent where I think in my masters’ system, configuration was probably a third of the whole system).
For the Game Tracker I set out to keep things really simple, and ideally to do things "the right way". Given that I don't actually work in software development with anyone other than myself, figuring out "the right way" can be a bit of a struggle sometimes. Still I'm trying to start modelling my own technology learning after Julia Even's approach and learn and focus on what I have learned.
For handling settings, I've used Java's Properties quite a bit (they made up a big portion of my PhD system, which was *much* more streamlined than my masters). Having an API to handle getting simple values in and out of a file is really useful. Preferences are an extra layer on top of that, where you hand over keeping track of the storage to the JVM.
While I'm still a little grouchy about Preferences not being transparent to the user or the programmer. They do feel like "the right way" to keep track of settings. I was especially confused when I cam back to the Game Tracker after a long break and so had to relearn a lot about Preferences. Hopefully this will be a good overview and collection of the good resources I found.
Overview
As I mentioned, preferences let you hand off worrying about where your information gets stored. All you have to do is tell Java that you would like this class’s preferences and then either read or store a value. When you end your program, Java will store everything and the next time you run your program your preferences will be there.
You can recompile or even completely delete and replace your program and your preferences will still be there. If you’ve ever deleted and then reinstalled a program and been surprised that it remembered everything from before, that’s preferences in action. (And that bit where stuff just gets magically remembered is why I’m sometimes uncomfortable with the idea of preferences, sometimes you just want a program to go away, but we’ll talk about that later.)
When we’re talking about preferences, we’re talking about the small pieces of information you need to make your program run. For example, with the game tracker the list of games is too big to go in preferences, but keeping the file name where I'm keeping the list of games makes sense. There’s not a strict limit, but keeping too much data in your preferences isn’t a great idea. Additionally, everything is handled as a string by Java so you want to avoid storing anything complex, like an object, and should keep in mind that you’ll have to handle moving numeric data back and forth.
Using Java Preferences
You access Java preferences through a java.util.prefs.Preferences object. We’ll talk about how you *get* a Preferences object shortly, but once you have an object you can treat it as a dictionary. You have a put(String key, String value) method where you can store a value under a key and a get(String key) method which gives you the last value you stored under key.
If you have other types of data, you have to handle them separately. The good news is that you don’t have to do any parsing, there are putBoolean(), putDouble(), putInt() methods and getBoolean(), getDouble(), getInt() methods (and actually there’s a pair of methods for every primitive Java type).
You get a Preferences object by asking the Preferences system for it. Similar to the way you ask System for System.out, there are static methods on the Preferences class for getting your Preferences object. Unlike System.out where there’s only one object to get back, there’s actually many possible objects.
Each package in your program gets its own preferences, so when you ask Preferences for your object, you need to give it your class so it knows which package’s preferences to give back. The static method Preferences.systemNodeForPackage(Class theClass), takes a class object and so you can statically give it a class like CLI.class. *
That there are actually two kinds of preferences. System Preferences (which is what you get if you use the method above) and User Preferences. System preferences are shared by every user on a computer, but the user preferences are unique for each user. Generally user preferences are the right place to store things unless you have a strong reason why everyone needs to share. To get the user preferences, you’ll use the static method Preferences.userNodeForPackage().
From there you’re pretty good to go. The first time someone runs your program you can store their preferences and there after you can access and update them as necessary. I’m not sure it’s the “right way” but I think you should also offer your users a chance to see what’s stored in preferences and a way to remove them (the Preference class has methods to do both). I still need to implement this in the Game Tracker, but I think transparency is good to keep in mind.
My GameTracker Example
The last time I worked on the Game Tracker I reorganized a lot of the components. At this point ** I’m really only keeping two preferences for Game Tracker; the name of the persistence manager and the file name of the file which has the game and play session data.
The persistence manager is managed in the CLI class. As the CLI class is starting it figures out which PersistenceManager it needs and then calls a factory to return an instance of that manager. For every preference, I keep a constant string in the class to keep the key value, and I’ll often also keep a default value as well.
When the CLI starts, it processes the command line arguments (a story for another day) and checks if the option for the persistence manager has been set. If the value has been set, the CLI saves the value in the preferences. Then the CLI consults the preferences for the name of the persistence manager (with the default). This means that the value that’s loaded is either the new value that was just passed in, the preference that was set before or the default value if it’s never been set.
private static final String PERSISTENCE_MANAGER_PREF = "gametracker.persistence_manager"; private static final String PERSISTENCE_MANAGER_DEFAULT = "CSV";
if (cl.hasOption("datamanager")) { Preferences.userNodeForPackage(CLI.class). put(PERSISTENCE_MANAGER_PREF, cl.getOptionValue("datamanager")); }
persistenceManager = PersistenceManagerFactory.getManager( Preferences.userNodeForPackage(CLI.class).get( PERSISTENCE_MANAGER_PREF, PERSISTENCE_MANAGER_DEFAULT), args);
Within the CSVPersistenceManager, the only preference kept is the data file name. Like the name of the persistence manager in the CLI, I keep the name of the preference and the default value as constants, although here the preference name actually sits in the super class FilePersistenceManager.
In FilePersistenceManager:
protected static final String DATA_FILE_PERF = "gametracker.datafile";
In CSVPersistenceManager:
private static final String DATA_FILE_DEFAULT = "gametracker.csv";
Persistence managers are passed the arguments off the command line, so as in the CLI, the super class checks the arguments if there’s a value for the datafile and that is taken and loaded into the preferences.
if (cl.hasOption("datafile")){ Preferences.userNodeForPackage( FilePersistenceManager.class).put( DATA_FILE_PERF, cl.getOptionValue("datafile")); }
The CSVPersistenceManager then checks the preferences to find the filename for the data file.
datafile = new File( Preferences.userNodeForPackage(FilePersistenceManager.class).get( DATA_FILE_PERF, DATA_FILE_DEFAULT));
The data file can be changed from the setFile method in FilePersistenceManager.
public void setFile(String fileName) { Preferences.userNodeForPackage( FilePersistenceManagerMenu.class).put(DATA_FILE_PERF, fileName); datafile = new File(fileName); }
I’m not getting a ton of use out of preferences right now. I was doing more before I started streamlining, but right at the moment those are the only two things I need to keep track of. Again, I’m not sure I’m “doing it right,” but this setup seems to work for me.
The Magic Inside
As I said, the great thing about preferences is that the JVM and the operating system really take care of it for you. So if you’re happy to trust the “magic system” to keep stuff up and running, then you’re good to go. On the other hand if you’re trying to figure out what the hell is going on, then let’s talk about it.
I ended up researching this because when I came back to the GameTracker after a long break, I had no idea what was going on or why the software (which I wrote) knew where the files were without me telling it (also sometimes I’m bad at taking notes - but I did have an issue on the topic for myself). This was especially a problem because I was looking for a local configuration file, which just wasn’t there.
I set out to figure out what was going on. Thankfully a number of people have written pretty good explanations of the preferences system. I decided that I could add a little bit by bringing those articles together and the best way to learn anything is always to teach it (or in this case blog about it).
So with that being said, there are two things to talk about if we want to open up the magic inside.
How are the preferences organized?
At the JVM level, all preferences are kept in trees. There’s the system tree and a tree for each user. (So as a programmer you have access to two trees, the system and the current user).
Each node in the tree represents a package. The root of the tree is the empty package and it’s named “/”. Then every package below the root is its own node. If we take my GameTracker as an example, the CLI.java main class is in package ca.kendon.gametracker.cli. That means that the root node has a child node, “ca/”, which has a child node “kendon/”, which has a child node “gametracker/”, which has a child node “cli/”.
A tree diagram of the CLI class's preference. |
Since we have multiple packages with preferences (.cli and .core) both of those packages's preferences would be held under their common parent. so "gametracker/" would be the parent to both both "cli/" and "core/"
A tree diagram showing the common ancestors for both the "cli/" and "core/" packages. |
Notice that the names of the preference node have those slashes at the end. If you ask for the full name of the node the CLI preferences are in it will be "/ca/kendon/gametracker/cli”. If that seems like a file name... well that might be important later (in some contexts).
Where are the preferences stored?
The JVM has its nice abstraction of preferences in trees, but that doesn’t help when your question is “where is that value *actually* stored?” Now we actually have to take a look under the JVM and see what the operating system is up to. Preferences exist in all operating systems, not just for Java but for any software that needs to remember things. Of course, since all operating systems have their own ideas about where stuff is, there’s a different answer for every operating system. For a quick overview we’ll look at Windows, MacOS and Linux.
Victor Kropp has a really good post on the details.
Windows 10
As windows is wont to do with all of its configuration information, things are stored in the registry.
User preferences are found at HKEY_CURRENT_USER\Software\JavaSoft\Prefs
System preferences are found HKEY_LOCAL_MACHINE\SOFTWARE\JavaSoft\Prefs
Remember that you can look at the registry using the Registry Editor (regedit) utility, which is installed by default in Windows.
MacOS
On MacOS, preferences are stored in the Library. Since we have two types of preferences, system and user, MacOS actually has two types of libraries (system and user) for preferences.
System preferences are found in /Library/Preferences/ in a file called com.apple.java.util.prefs.plist
Each user’s preferences are found in their home directory in ~/Library/Preferences/ in a file *also* called com.apple.java.util.prefs.plist
plists are Apple’s favourite way to store things in a way that slightly frustrates everyone … well it frustrates me certainly. They’re XML files (often compressed) which can be a pain to actually look at. On the command line you can run the command:
defaults read ~/Library/Preferences/com.apple.java.util.prefs.plist
Which will show you the preferences for your own user. For example mine includes:
"gametracker/" = { "cli/" = { "gametracker.persistence_manager" = "CSV"; }; "core/" = {
"data_file" = "data/games.data";}; };
If this seems a little overbuilt it's the weird collision of Apple's plists and xml files which always end up having some weird structures.
Linux
On Linux, preferences for each node are stored in its own file (in case you’d wondered why the Java name for preferences looked so much like a file name).
System Preferences are stored starting in the directory /etc/.java/.systemPrefs/
User preferences are stored in each user’s home directory under ~/.java/.systemPrefs/
Under those directories, the JVM builds a directory structure that matches the preference nodes, with the leaf nodes being files. In my Game Tracker Example, that means that the "gametracker.persistence_manager" = "CSV" preference is stored in the file ~/.java/.systemPrefs/ca/kendon/gametracker/cli.
This makes a lot of sense to my *nix-centric brain and I think explains a lot about the structure they chose when setting up preferences in *nix environments.
Conclusion
I'm always wary of systems that aren't transparent to the user, but preferences provide a good way to keep track of the details about a program without making a mess or requiring a lot of overhead. I still have a lot to learn, already, I'm wondering about some of the things I'm doing to update the preferences from the program. Hopefully this overview provides a place to get started and answers a few of the questions the preferences seem to draw.
Footnotes
* I *think* you can use the this identifier to get the class dynamically, but I'm not sure whether or not that's a bad idea. (Thus far I've found I've made mistakes using the literal class - copy and paste errors - so using this may be smarter.)
** This is a snapshot of the GameTracker from early November 2021.
Sunday, January 09, 2022
Friday, January 07, 2022
Tuesday, January 04, 2022
2021 in Games
I've now spend 5 years keeping track of the video games I've played and I've found it lets me play more and play more intentionally. Plus it's nice to be able to write a post like this where I can look back over what I played, what I enjoyed and how the year went.
Yes, it was mostly Battle Brothers |
Time Spent
I played 49 games for 600 hours in 2021. That's much less time than I've spent on games for the last few years. I guess that makes sense working full time and (slowly) starting to build up my creative career. My PC also broke a little and I think that cut down on some of what I played, and it also forced me to be more intentional with what I play. I also dedicated a lot of my playing time this year to trying out games from the cellar that I hadn't played for one reason or another.
The curve on how long I played games is pretty sharp. I played 150 hours of Battle Brothers and would have played more if the PC hadn't started to fall apart. I feel like I spend a lot of time here writing about I like tactics games but I'm not good at them. Battle Brother's hasn't really changed that for me, but it has been really fun to play.
Everything else topped out at around 50 hours, so overall my top 10 games by time played where:
- Battle Brothers - 150 hours
- Fire Emblem: Radiant Dawn - 59 hours
- Chrono Cross - 47 hours
- Trials of Mana (Remake) - 41 hours
- Mario Golf: Super Rush - 40 hours
- Hyrule Warriors: Age of Calamity - 37 hours
- Fire Emblem: Path of Radiance - 36 hours
- Trials of Mana (Collection of Mana) - 32 hours
- Legend of Zelda: A Link to the Past - 24 hours
- Secret of Mana (Collection of Mana) - 22 hours
After that everything flattens out into pretty small chunks.
I really wanted to *finish* a lot of games this year, and I did, but I think I really want to *experience* a lot of games for 2022 and also focus on having as much fun when I'm playing video games as possible.
I love a good end screen. |
Finished Games
I haven't kept track of the games I've finished before, but having spent more time in and around ProtonJon's community I thought it would be interesting to see what I've finished. As I said, I already thing my goal of 2022 isn't to finish as much as it is to play, but it's still interesting to see what I played to the end.
I'm counting finished as reaching the credits at the end of the story, although for practically all of these games there's a lot more to play at that point. More or less in the order I finished them, I played:
- Hyrule Warriors: Age of Calamity
- Super Mario 3D World + Bowser's Fury
- Fire Emblem: Path of Radiance
- Fire Emblem: Radiant Dawn
- Trials of Mana (Remake)
- Trials of Mana (Collection of Mana)
- Celeste
- Legend of Zelda: A Link to the Past
- Super Mario Odyssey
- Secret of Mana (Collection of Mana)
- Chrono Cross
The only game I really completed was Bowser's Fury. It was really well designed to make it easy to pick up and play for a few minutes and get a few stars. For the others it was mostly nice to see the end of the story.
If you want a game full of cat's Cat World certainly is. Still not sure why they write it Bowser's Fury. |
Favourite Games
Every game
I think the games I liked the most were (listed alphabetically):
- Battle Brothers
- Hades
- Super Mario Odyssey
- XCOM: Chimera Squad
When you do a thing right, Battle Brothers feels pretty good. |
Things About Games in 2021
As I said in my Games of 2021 post, everything I played was good, but I don't think 2021 really hit the heights of 2020. I spent a lot of time trying to get really deep with a lot of the games I played and I'm not sure it was really worth while. I'm glad I played the games I did this year, but looking back on what where my favourites out of what I played, the list is weird. I only played a few minutes of Hades and Chimera Squad and not that many hours of Odyssey.
Battle Brothers is great and if I can manage the multi-headed hydra of fixing my PC (or more accurately, figuring out *how* I want to fix my PC) I'll be back to it. I think it's a great game where commitment, to characters and attempts, is really well balanced with interesting outcomes. I *like* Dwarf Fortress, but it always feels like it takes too much commitment for the "things went really sideways" fun to kick in. With Battle Brothers I've screwed something up in an interesting way after a few hours, or if things get boring, it's easy to jump out and start again.
I also like that I get a little better every time I play. I keep an eye on the Battle Brothers Reddit and sometimes I feel like I'm not getting enough from the game, or playing it right, but I'm having fun and that's really all that matters.
I'm trying to "Do" more in 2022 and I think that's going to apply to games as well. I've said it a few times, but I'd like to play more games and play more types of games. 2021 was a little bit stayed and I think I forgot to have as much fun as I can.
I recently read a tweet about how you shouldn't consider games you want to play but haven't a "pile of shame" so much as the video game equivalent of a wine cellar, where you're waiting to find the right moment and mood to open something up and I like the reframing. I think I can add that games don't go off after your start them, so there's no worry about putting stuff down and picking them up again later and you may as well have as many open games in your cellar so you can find everything you enjoy.
Saturday, January 01, 2022
Games of December 2021
December was a bit weird, but I feel like trying to expect anything from a month during the pandemic is kinda useless. I ended up mostly focused on Chrono Cross, which was find, but next year I really want to play more and more kinds of games.
My top five games (by play time) for December were:
- Chrono Cross - Chrono Cross is fine. I decided to play along with the Axe of the Blood God game club and found I enjoyed the minute to minute play, but was pretty underwhelmed by the story telling and most of the mechanics. As a sequel to one of my favourite games, Chrono Trigger, it feels deeply up its own ass in story telling. Chrono Trigger is able to get by, by not spending a lot of time integrating the plot or the time travel, Chrono Cross spends so much talking about time travel that I was just hoping to find a portal to the end of the game.
The combat system is also underwhelming, partly because they tried to make it so complicated. At the end of the day most of the really fancy things they implemented didn't really work. I think if they'd leaned into the puzzle colour grid and made that a much more intentional system, rather than frustrated by the random AI. I think if they'd mixed this with Final Fantasy X's combat system, it would have worked a lot better.
I love how colourful Chrono Cross is and I love the island aesthetic. It would have been nice to really get to enjoy the world a little bit more with less of the story sitting on top. A lot of the time it feels like they made a game and only later tried to tie it to Chrono Trigger, and I'd love to play that version that's a little more laid back and embracing the weirdness.
- Secret of Mana (Collection of Mana) - Had a little bit of Secret of Mana left, so started the month by finishing that up. I ended up running out of steam fighting the final boss and ended up with 1 MP left, so had to sit there and wait for the game to knock me out. I ended up doing my old fashioned magic grinding to end the game off. Stupid as it seems standing in an inn and casting magic on myself for a few hours is still kinda fun.
- Super Mario Odyssey - Really didn't pick this up much, but played some Balloon World and really enjoyed it. Odyssey just has a great feeling for movement.
- Legend of Zelda: A Link to the Past - Went back into this a bit, but mostly I'm planning to spend more time in 2022 learning the game and getting good at it.
- Celeste - Played a few minutes just to get the feeling.
The Books I Read - December 2024
The Lays of the Hearth Fire series by Victoria Goddard is good. It's also long, but I really enjoyed spending most of December on ...
-
A quick note, I've updated this project ! I don't usually go for having a lot of stuff on your desk, but my recent success makin...
-
This volume of my favorite YouTube videos is a bit of a testament to the cool things people can make. So have I mentioned that George Wa...
-
One of my first memories in our house, about 7 years ago now, is sitting at the dinning room table and thinking the fridge was about to expl...