Rails: The Missing Parts
Radoslav Stankov • Sofia, Bulgaria • Talk

Date: December 07, 2024
Published: December 15, 2024
Announced: unknown

Radoslav Stankov was the CTO of Product Hunt. He is currently building Angry Building (https://angrybuilding.com/) as the single technical person on the team. This is his Rails stack.

Learn about the flavor of Rails that Rado likes and uses.

Ruby Banitsa Conf 2024

00:00:07.480 in this uh how to say like scenery like such a it's a very Co weather we just
00:00:13.880 need like a fireplace with like maybe we had like a new monitor we just having a a fireplace on the side making it even
00:00:20.160 more cozy especially with the bitas the traditional food to eat in this cozy weather so yeah so my name is R uh you
00:00:28.599 can find me everywhere are as rust tank of everywhere I have like a substack
00:00:35.120 called tips where I share some random tips and because again there is a lot of
00:00:40.680 food here and uh always people make photos of the slides and I change slides
00:00:46.440 very fast and I show a lot of presentation so all my code from all my slides are already listed here so you
00:00:53.520 don't need to kind of frush your phones and just uh take photos there will be a lot of code because I like doing Cod
00:01:00.399 when I can so yeah today uh so if we talk about rails there is this slogan
00:01:05.880 one person framework and uh right now I'm the CTO the founder and the only engineer for
00:01:12.200 the next 12 days uh we have some new people
00:01:17.960 coming in but uh I buil this startup it's going rebuilding and basically uh
00:01:24.880 I'm one developer I build the whole system I build the mobile app for iOS Android and basically a big chunk of
00:01:32.399 that was because rail really is the one person framework like honestly again without rails I would have spent a lot
00:01:39.280 more time and coffee and unhealthy food but yeah Ruby on Rails is like a great
00:01:45.000 tool it's like an amazing thing to build and often again when we choose Technologies when we choose things uh
00:01:53.600 when you need to build stuff I'm always like asking when people want to build stuff should I do something new should I do something C and I'm like
00:02:00.560 uh first again if you want to build a product use something you know in without the only exception is if the
00:02:07.719 thing you know cannot solve the particular problem of the things you want to build if there is like unique technologies that enable stuff because
00:02:14.239 one problem with the new exciting Tech is there is a lot of fail scenarios which you don't want to do like for
00:02:20.440 example should I use this other cloud service uh I don't know does it have something better I know o AWS books and
00:02:29.200 I know how to work around them that's very important so yeah so we all know rails rails has a lot of things included
00:02:36.800 in in in the Box a lot of packages a lot of packages that people actually this is like if you open GitHub this is all the
00:02:44.200 rails gems with all the stuff that come here surprisingly some of that stuff I
00:02:49.400 have never used in an actual project uh like the M the the thing which receives
00:02:55.360 emails and process them and do all the magic with them for example something very useful but but it has a lot of
00:03:00.840 things and again if we open like a new rails project we see a lot of like things being included a lot of batteries
00:03:07.440 included and that's amazing that's great usually when I come here I just like comment out some of the stuff I don't
00:03:14.000 need to change some defaults but in general rails and the EOS system has a lot of B pieces and again it helps you
00:03:21.360 build a lot of like nice apps and out of architectures but there is also one part of rails where it's still again every
00:03:29.040 project is different every project kind of has to build a lot of like its own combining the pieces but also there is
00:03:36.599 some things that are in my opinion missing that this is something which I noticed like from a from a lot of other
00:03:42.599 projects I always have those places which I just copy paste from Project to project some settings some things and
00:03:51.000 it's always a lot about like uh the funny thing is I have this photo and the
00:03:56.319 caption is R laughes active record and does this look somebody who loves that something I'm not very photog gentic
00:04:03.000 here but yeah uh in general again I really like Ruben royos is like one of the best things and because when you
00:04:08.319 love something and you become an expert with it you know all the things it's missing so today I'm going to have two
00:04:15.319 things I'm going to share like one things is rail extensions like things
00:04:21.280 which I basically copy paste from Project to project to project to project to project and some missing Concepts
00:04:28.280 like things that there are still not not in rails and it's stuff that I have seen in a lot of projects people just invent
00:04:34.639 the things around those things and one one one funny thing was
00:04:41.240 when I was making the rails extension part I'm like oh this is Ping from rails and I'm like maybe I look at the
00:04:47.520 documentation and they see actually rail had this thing so I found some features with like shallow shallow routes and
00:04:54.560 some parameters that were actually I didn't know they existed so the things I'm going to talk
00:05:00.440 today is like active record routing Arrow handling and internalization and again some of those
00:05:08.160 things are basically things I put in every project like the first thing for active record and this is honestly every
00:05:14.039 time I go to a real project even if I don't own it I add this gem it's called
00:05:19.360 annotate which basically puts this comment in front of every active record
00:05:24.479 model and it's honestly I have no idea why it's not in rails or ar doesn't have
00:05:30.039 something like that that it doesn't have it
00:05:35.919 builtin somebody doesn't like yeah I that's a personal opinion about all those stuff uh the thing about it is a
00:05:43.440 lot of the times you open a model and you have you don't know what are the fields you have to for example go to the
00:05:49.600 schema often what are the indexes what are the foring keys like a lot of the
00:05:54.680 stuff for me this gem is basically a must like yeah some people don't like it
00:06:00.039 some people yeah but I don't want to go to
00:06:06.599 like the schema because the schema is an sqo file and when the schema oh the way you can ask questions you can disturb it
00:06:14.199 because it's a lot of like discussion Parts but yeah I mean it's the thing but for example that
00:06:20.400 that's why we added it because we have schema SQL with very complex SPO queries and you always have to jump between two
00:06:26.240 three four five six FES and having this your file it was actually very useful
00:06:31.840 and also when you see it like that it's very obvious for example you notice bugs here like oh something here is not null
00:06:38.599 like why account ID and user ID in this thing is null like why building user ID
00:06:44.479 is null here what's the reason and you notice a lot of things when it's like right that another thing which raos has
00:06:51.360 it built in and it's amazing and I'm a big fan of counter caches raos has a buil-in counter cache which is great but
00:06:59.240 it's great for like when you have a very clear so what counter caches does first
00:07:06.240 is when you have a column which counts how many records of some relationship
00:07:12.120 are in the project so in in general rail terms you only count the whole
00:07:17.680 relationship but often for example let's say you have a user and oh actually here
00:07:23.800 I have an apartment apartment belongs to building and some of the apartments are archived apartment department so you
00:07:30.240 want to have a separate counter for that and this G Counter Culture what I like
00:07:35.840 about it is allows you to add a lot more complicated counter cache things so The
00:07:42.400 Counter Culture gem is basically an extension to the counter caches and it
00:07:47.639 works amazingly well especially again if you have an application that does a lot of calculations and it also uses counter
00:07:54.440 caches not only for counting columns but it can also do things like okay
00:08:00.080 do a counter cash based on the sum of all transactions so you don't need to write a lot of the code so again real
00:08:06.199 has this as a concept but this is a gy I basically add by default in every project another active record concept
00:08:14.400 which I personally copy paste from every project is this thing which is called belong to
00:08:20.240 polymorphic so raos has polymorphic associations and again they are controversial some cases you use them in
00:08:26.639 some cases you don't it's very useful when you need to attach things to other things like let's say you have comments
00:08:32.440 and you want to allow a lot of Records on your system to be commentable or in
00:08:38.039 my case I have a note and the note can be attached to many things one problem with the default rail implementation is
00:08:43.760 if you just say belongs to polymorphic true you actually don't have any information about what are all the
00:08:50.360 possible types that I can actually have this polation being part of it's a lot
00:08:57.320 of like chasing uh uh what is this and yeah you can do go in the production database and
00:09:03.519 list all the changes because most probably locally you won't have everything but having like just
00:09:09.480 basically belongs what it does is basically validates that those are all the classes that have nodes to and I
00:09:16.760 have found that in like bigger projects when you don't exactly know what happens
00:09:22.320 it's very it's a lot easier to know oh this is what this things can have also
00:09:28.760 there is a con in rails which it's called delegated types how many of you have used delegated
00:09:34.839 types yeah it's it's kind of similar the problem the thing with delegated types
00:09:40.040 it's the reverse Association like delegated type basically tells you my type is this and I delegate to this
00:09:46.040 while polymorphic is I own this so delegated types actually have the same
00:09:51.440 concept like over oops sorry
00:09:57.640 yeah five tickets yes yeah so uh delegated types I was very surprised but
00:10:04.560 also delegated types actually had this things so it's not something I can imagine being in real some days that we
00:10:10.440 say oh this polymorph Association allows you only these types um and again another thing which I do in
00:10:19.600 my projects which is I have never seen somewhere else so we know how in rails
00:10:25.600 we have the concerns when it's like include concern and concerns are
00:10:31.320 trade if you don't overuse them uh but I found I found them a bit limiting
00:10:37.600 because imagine you have a concern like uh has audit log for example imagine
00:10:45.079 this is a concern and I say this issues activity has an audit and I want to get
00:10:51.160 all the fields except this one so having it with a regular concern you need to
00:10:57.440 define the static method in the concern then you have to keep that state you need to call this method you have to
00:11:03.079 remember to call this method for example I have this extension it's
00:11:08.440 called angry support time as Boolean like for example I have a extension which like this field indicates that I
00:11:15.279 have a Boolean field which takes its value from like a time at coln and if I
00:11:21.200 have to do this with concerns I need to do a lot of like meta programming a lot of like need to Define like a static
00:11:27.000 method then I have to include the concern they need to do and concerns work amazingly well when you need don't
00:11:32.720 need to configure them like like let's say this modification subject is very in easy thing I can do I can say include
00:11:39.839 notification subject and that works out of the box all settings and all of that so again this is very kind of like
00:11:47.880 concerns uh but what I do with my um with my extensions the way I do them
00:11:54.920 from basically every record is my extensions just just have one method
00:12:00.040 which is called defined and it receives a model it receives its settings and it does all the magic that's kind of needed
00:12:06.000 for its record so it's done basically it decorates the record it also works as uh
00:12:14.880 if you noticed it also works this I here is not so visible but also what we have
00:12:21.480 some extensions which are basically concerns yeah yeah basically everything we I don't use in my project I don't use
00:12:27.519 concerns I just use those extensions because EXT I will show you the call of
00:12:32.839 the extension but this basically allows me to say okay this is the model this is the field and this is basically what they need to do for like BR field like
00:12:40.199 there is a lot of code here imagine doing this with like concern you have to do a lot of this orchestration yourself
00:12:46.160 and the user who the user who uses your concern has to call those methods by themselves they have to know what are
00:12:52.240 the methods what the conventions and they have to do a lot of work and you need to add a lot of methods to the
00:12:59.760 whole model and the implementation here is very simple you just say extension
00:13:05.600 the extension module just call def def if there is if there is not you just
00:13:11.120 basically use it as a concern so every concern actually go can be also an extension so this is a small trick I
00:13:17.440 kind of use in all my projects and it helped me a lot with exactly those concerns which are more like oh this is
00:13:23.399 like a small Library here like like for example have an audit loog and just say okay don't I I care about all the fields
00:13:29.720 except those ones so this is something I haven't seen anywhere else done so most probably I'm
00:13:36.440 totally wrong when doing it but it work quite well in like five six places where
00:13:41.959 I introduce those things uh routing so rails routing I really like
00:13:48.600 the rails routing in general it's great also rails has this concept which
00:13:53.720 honestly JavaScript World Is Still discovering it you actually can have methods that you call and generate the
00:13:59.959 route and the methods and the routes are different so you actually can have a nice routing methods and a nice URLs and
00:14:07.279 you don't need to remember routes but with routes there is one thing which is always missing I want to use my route
00:14:13.600 helpers outside of the mailer and the action pack this is something in every project I need some way to get the
00:14:20.720 routes for example I want to do a push notification I want to send an SMS I want to do uh I don't know just use the
00:14:28.519 routing outside of my main life cycle of the method and rails actually has that
00:14:34.519 you can do rails routes and do it and then it blows up because you don't have the default URL options because it
00:14:41.800 doesn't know the domain it doesn't know a lot of that so in every project I I basically have this class it's called
00:14:48.120 routes I include these rail URL helpers I set the default options and everything
00:14:54.399 works I would focus on this routes customing a bit so sometimes there is in
00:14:59.480 rails there is this awesome method it's called URL 4 which it gets a method and
00:15:04.560 like a class and it and it gets its route but the problem is if your routes
00:15:09.600 are named differently than the models if you have Nam spacings if you have like that so also one thing with rails routes
00:15:18.320 is sometimes I actually need custom routes like imagine you have like a route which accepts a build like in my
00:15:25.240 case I have an account accounts have buildings and I can filter things by that so I want to say I give you a
00:15:31.120 building and you give me like the transactions of this building and it it takes the account of the building for
00:15:38.440 making the route I want to have like a custom route method and in raos having a custom route method where do you add it
00:15:45.399 how where does it get into the system so what I do is I have this routes custom I default all the routes there they're
00:15:51.839 included in all the places where the routing is included and also in this rout helper so in this way there is no
00:15:58.120 difference between my custom routes and rails routes and for example instead of URL
00:16:04.079 full I have something which I call the record pth and I do things like that there which not that pretty but it's
00:16:09.839 very useful in my system because I have a very clear mapping between oh this is my CRA this is my models this is my
00:16:17.560 methods and it down fors to R for it it I found this very useful because it
00:16:23.639 clean up a lot of my routing places because if you notice some of my routes are very long and also having this P
00:16:33.279 record and using the URL for in general cleans up a lot of like you can make a
00:16:38.519 lot of generic helpers instead of just passing those strings so this is another technique I
00:16:44.560 have found in a lot of places so another place in rails which
00:16:50.560 rails has a lot of building tools and it's very great it's Error infrastructure and why it's Error
00:16:57.120 infrastructure is great is because a lot of gems can actually plug in into rails and get all this data and do a lot of
00:17:03.880 that parts for you like you can put something like Sentry you install it with two clicks or New Relic or whatever
00:17:11.679 and you get all the error back however I have found that in all the project I make a custom module all named error
00:17:18.480 reporting because every error reporting framework is it Sentry is it hub spot is
00:17:25.480 it signal whatever needs things like okay I need to assign a user like to say
00:17:30.919 in my controller where my user logs in this is the user who this error is for
00:17:36.520 because often when you see an errow you want to know who the user is you need to get this data also I have used couple of
00:17:44.280 other helpers here like uh capture exception and exception uh context like
00:17:50.280 for example often I need to be able to capture an exception in my error tracking and for the user they don't
00:17:56.799 need to see that exception like sometimes when I write code and I get some error State I use this kind of like
00:18:03.039 a more advanced lock so having this as like a facade around the service like
00:18:09.799 for example if I want to use something else in the future I can just change it also there is some useful things like
00:18:15.799 for example with active job uh if a job is
00:18:21.559 discarded uh like for example it was to Tred too many time you call discarded via some helper or something around it
00:18:29.000 there isn't like in rails as far as I know none of the exception trackers by default track all the discarded jobs
00:18:35.080 where did they come from there is a thing there but you need to like tune it very slow so I also have this explicitly
00:18:41.720 called in my Base active job if the job is discounted please tell me about it also again there is a lot I can talk
00:18:49.200 about exception some self-promotion I have this talk with videos about it's
00:18:54.280 called living without exceptions it's a lot more Deep dive on how to have
00:19:00.039 exceptions so internalization having like internalization tools in rails they're
00:19:07.039 actually amazing like I'm very happy about them the they are a lot into the
00:19:12.280 whole core of the system you can basically have a lot also internalization in Ruby in general and
00:19:19.880 this is very unique to Ruby because I have tried to do internation in other languages Ruby has this magic thing
00:19:27.760 called a symbol which is different than a string and there is and in other
00:19:32.840 Frameworks they only have strings like JavaScript and all the things so with Ruby what we can do in a lot of places
00:19:38.960 is if you pass a symbol the symbol can be treated in a different way so for example I have this helpers which I
00:19:46.000 inject directly into my E e88 help one is called T display and T
00:19:52.520 display is basically a think which accepts another thing and give you translations of it so if I get a symbol
00:20:01.159 and that's the magic of Ruby that's honestly not not other Technologies can do something that clean if I give you a
00:20:06.840 symbol just translate it just go to the translation table translate me my string
00:20:12.360 works perfect if it's a string okay just display it so if if this is the username
00:20:17.640 if I give you a string just translate it uh apart from that I have my own tool
00:20:24.320 link around uh to display if this is like a record if it has a display name just use
00:20:30.600 a translated display if it's a name name title ra has some ways to do like
00:20:36.840 default naming of Records but they don't have one which is explicitly used for
00:20:43.000 translations but this is something I have found when you do translations in a project having AER like this cleans up a
00:20:50.360 lot of noise uh and a lot of the times you actually want to say oh I I I know what the text needs to be and sometimes
00:20:56.880 you want to do the key and this this part is particularly R magic because none any other technology has like
00:21:02.679 symbols like again some El here like but for example JavaScript sft and other
00:21:08.320 things don't don't have that another two helpers which I find very useful my is
00:21:13.720 two options which is basically you give it a collection of records and it just returns them to display them so
00:21:19.960 basically I can say oh here is records give me all their translations in a list
00:21:25.360 and how they are displayed and I have t which re receives an in from like
00:21:30.720 imagine you have a drop down when you pick the status of a product or state you just okay here's my keys and I just
00:21:38.000 guess the key name based on that internalization and also this clean up a lot of the
00:21:43.919 stuff another thing this is more particular for my project I have those two helper methods which I created one
00:21:51.080 goes to all the translations and matches all the files if there is missing strings in them like if I have a
00:21:58.000 transation in Bulgarian but I don't have it in English this will blow up and this goes to open Ai and just fills up the
00:22:04.360 missing Keys which works surprisingly well like I was able like I didn't know
00:22:10.919 Spanish I needed to translate like 3,000 keys in Spanish and then I give it to
00:22:16.880 somebody who understands my domain and understands Spanish and he only changed like three words and the only reason he
00:22:22.960 changed him is like yeah that's correct but in this domain this is a bit more specific and actually this word quite
00:22:29.039 well like just when you have like translations in your project just go W them through okay my CI ures I don't
00:22:36.039 have missing keys and if I have missing Keys okay just go to something that translates
00:22:43.159 them all right yeah this is some some tips around internalization if you care
00:22:48.640 about it like uh because I did a lot of work with internalization the last few years always raise erals in development
00:22:55.720 Only log them in production uh don't I I don't like using the nested keys with
00:23:02.840 rails like in your translations you know you can have like nested Keys you can just C them by name which work well when
00:23:09.440 you have like a framework like rails that can Nest errors but I have found that it's a lot more simpler for me just
00:23:16.960 to have those symbol underscore divided places it's easier to parse the found
00:23:23.200 sort it find the missing keys and honestly most of the time when we use that do use dot here or use underscore
00:23:30.559 it doesn't matter that much but it makes it a lot faster to work with also again
00:23:36.080 use defaults like a lot of the times which I do in my project is oh I have a table column like I have my tables and I
00:23:43.440 say the table column name is this but if the table column does it's not translated maybe it's a form label
00:23:49.120 because most probably it was entered by a form so I use a lot of those
00:23:54.480 defaults so that was like some of the extension part of the Rails any questions
00:24:00.640 comments Kate from Gennady show youor oh that would be fun is it 20
00:24:07.919 Lines no no no all right so some of the missing
00:24:15.120 concept so this is things that we basically have in every project so we
00:24:20.880 need to have some UI view things we need to deal with external Services we need
00:24:26.640 to have some kind of like public and things that are permissions we need to do the service
00:24:32.600 object form CER object so the view components like this is something missing in rails it almost got into
00:24:40.200 rails but long time ago remember remember the talk from balcon Ruby about
00:24:47.760 it yeah but view component basically is a tool coming from GitHub which is like you can basically use classes to Define
00:24:55.000 your um your helper like basically Define your UI instead of using helpers
00:25:01.720 because with helpers there is a lot of mess around them and with partials there is even more mess even with the new
00:25:08.159 thing with the partials when the partial actually know yeah the strict Locos of the paral is amazing but they still have
00:25:15.039 those problems and having view components allows you to just Define
00:25:20.480 classes very simple classes and also they have this awesome plug-in which is the the main the other reason to have
00:25:29.360 this is the the same way in rail mailers have the mail previews which are very
00:25:34.399 easy to do there is these view view components previews that you can
00:25:39.679 basically have something like this and again having all your view components
00:25:45.440 being able to be separated and developed in Works quite well and do you use view component do you use flex components or
00:25:52.440 anything else doesn't matter in my opinion the question is it's really nice to have a class with some local logic
00:25:59.919 with private methods to do some complicated things and be able to be kind of
00:26:05.600 testable in separation I like and I can talk a lot about them and I have talked a lot about them here is a talk from I
00:26:12.520 think Ruby bannet a couple of years ago about view components and I also have like a whole
00:26:18.600 post about it so let's go to another thing external services this is
00:26:24.000 something we all hate and we all need to doal about it and the other thing is I have to it every mature El project uses
00:26:30.760 every popular and not other popular networking gem this is something which is like why we have two 100 networking
00:26:37.520 gems and I know why everybody like dislikes something about them and it's
00:26:43.000 like all right we have them and there is so much about them and in my opinion
00:26:49.880 this something again rails like before active jobs was in rails we had a lot of
00:26:56.559 like job related things things we didn't have a common interface and I'm surprised rails haven't done anything
00:27:02.640 with like oh there is active networking or something like that
00:27:09.640 where yeah I I mean think about it 2002 building part of rails which you can
00:27:15.080 actually depend also when we deal with networks this is something this trick
00:27:21.159 actually genady taught me long long long time ago in a galaxy far far away in
00:27:26.720 Germany we had is I have this module when handling errors which in so I have
00:27:34.640 this module called handle network errors and I don't know if you know when in Ruby you do uh TR catch blocks uh if you
00:27:44.399 pass this module as the Handler it's going to call the triple equals method of the class so that's how you can
00:27:50.200 actually match if an error matches the the error thing so I have this handle network errors and I basically have
00:27:56.760 listed all the exceptions of all the libraries that are included in my project so basically I have like 200
00:28:04.159 error here because they're all kind of the same in a lot of the times you can again split some some of the time out
00:28:09.960 some of the 500 but you just put this here and you can have like a single way to handle all the errors this is
00:28:15.919 something I do right now in a lot of my projects I also a lot of times I actually use this with active job where
00:28:22.919 in all my active jobs if there is like a network error do like five retries
00:28:28.200 otherwise discard the job lock it and so on and so forth one great thing with rails which
00:28:35.679 I'm so happy they added this a long time ago is the secure credentials in the yo
00:28:42.760 file so this is amazingly having this because before that we need to juggle
00:28:47.840 environment variables or we needed to do a lot of like the key security like K KS
00:28:55.120 and other Amazon sounding weird things so this is always great I always like
00:29:00.679 for example when I do Services I always have that keys in my system I have the
00:29:07.600 service name the token the secret and I always have this comment by the way please every time you add something to
00:29:13.000 this file if you learn anything from this stock at this comment this key is taken from this URL I especially if it's
00:29:20.679 a Google service with Google services you have no idea where this k key comes from and every time I have a poqu W and
00:29:28.360 I see this file ex change I go in and I beat people with a head if they haven't had this where do you get this key from
00:29:35.720 because I cannot tell you how many times some developer on the team registered with their company account to some
00:29:42.799 service and they didn't use the company account of the service so when they leave and their service expires the keys
00:29:51.000 uh blows up and we're like where did this key come from especially now when people just randomly register for open a
00:29:57.760 keys that's a problem to have that there another thing I do to battle the
00:30:03.600 external Services stuff is everything which I do in my projects regarding
00:30:09.120 external Services is grouped into its own namespace it's always called external it's namespace so I know if I
00:30:17.840 call it that means Network external is always a network called I have this template it's always like a
00:30:25.760 documentation like okay this is the server this is the URL of the service this is where do we manage tokens from
00:30:32.080 this is the API documentation if we are using a jam this is the jam that's being crapped by this service the idea here is
00:30:39.760 if you have 200 of those Services it's very hard to know details around stuff
00:30:46.279 also I have it like my favorite extent cell so it's this is just like a facad
00:30:52.760 in front of all the external services and this facade based basically
00:30:58.639 always have documentation around okay we are calling this method from this documentation with this area and here is
00:31:04.279 an example with unsplash like unsplash is a service for images so in a project
00:31:10.559 we needed to have with unsplash two things search and tracking download
00:31:15.720 because they want you to track uh the downloads and this is how this look like
00:31:20.840 I was like okay my services and splash this is the documentation this is the gem we are
00:31:26.200 using this is the port portal where we get our o tokens and this is
00:31:31.720 documentation here again this is the wrap of the URLs here we also handle the
00:31:36.960 errors like we hide all those networking errors regarding this and for the developer they just use this and every
00:31:43.519 time you want to say oh I just need this one search place in one product in my
00:31:48.960 application uh that's a problem because you again you don't know what are the external calls your application does
00:31:55.279 another funny thing was on a project uh I helped with they didn't have that
00:32:02.000 and they needed to block all external network Calles from the fire rall of their like pots and they were like do we
00:32:09.039 call this service and they had no idea which were the upcoming networking calls and if you have something like this it's
00:32:15.039 very easy just go to one folder and just go next next next next next I actually have a blog post about it with like a
00:32:21.080 long dealing more with external services so networking again is something where
00:32:27.240 again rail has some great fundamentals but we need a lot of the times people just okay I just run this gem I'll just
00:32:33.639 include it and we use it directly and the main thing in my opinion is never
00:32:38.960 just use the gems directly just WRA them so if they change versions if they
00:32:44.279 change apis becomes very hard like something like stripe for example uh
00:32:50.440 yeah it's very big action point but it's really good if you just had all the networking which
00:32:56.559 you don't own in its own like bucket even if it's a lot more C if you think
00:33:01.799 why should I make two codes here it makes sense when you when they break API or you need to update another thing how
00:33:10.080 with time 20 minutes okay I'm done in five I'm joking I'm not
00:33:17.919 so another thing we have something uh it's policy objects like raos right now
00:33:23.840 have this awesome Tools around how how you can make your own uh logging system but what one one area
00:33:31.720 which kind of is still missing is how do we authorize how do we handle the authorization in like a rails project
00:33:38.039 every project kind of like thinks about like the authorization themselves right
00:33:43.080 now I think the most popular gem is pundit pundit pundit what that's I think
00:33:49.840 I personally use a project because pundy didn't exist like 10 15 years ago I had
00:33:56.080 a gem called Kitty policy which is my own version of like policy files and again this is something I
00:34:04.440 personally use project to project to project because kit policy is just a GMI road to build authorization system
00:34:11.440 particular for each project and kit policy is very simple if you don't use Kitty policy you just write something
00:34:17.720 like this this is how Kitt policy started we have a file which was called application policy and this is like all
00:34:23.720 the permissions our system needed to apply it used my favorite extend self method with like static methods which is
00:34:31.639 like can access account user account if the user is null return false if the user is admin or if it's account member
00:34:38.560 the user like having this file with like 200 methods this this basically is the
00:34:45.119 authorization system which I found very useful because all the authorization was in central place it was big it was
00:34:52.159 obvious it was easy to test it didn't have any state it didn't do any
00:34:57.800 so this does zero allocations uh because one thing if you have like something which I don't like
00:35:03.560 about pundit is you need to insatiate a lot of classes and you do a lot of gu work with class names with like a long
00:35:10.680 structures and if you have like a graph Q API where every field you have to
00:35:16.400 authorization check if the user can see this field or they cannot it becomes
00:35:21.560 really a lot of like oh lookups and things like that with here again you have zero can just say okay just
00:35:28.440 guess me a new method name and just call it and um what quity policy does is
00:35:33.880 basically this method here becomes something like this it removes some of the noise like it
00:35:40.240 doesn't need to check if the user is a guest it can say okay this is the policy this is the method it basically
00:35:46.320 generates the same name method here it just got like a small DSL to do that basically skipping me this part and make
00:35:54.319 it more consistent the naming so I can do do things like can a user action
00:36:01.160 whatever can out this is a Boolean operation so this returns to for all
00:36:06.280 this blows up where the user cannot access certain things so this Rises like
00:36:12.000 and this is basically what Kitty policy is on two level it's like 20 lines the
00:36:18.480 thing about it is in my current startup we have uh a very expensive permission
00:36:26.079 system with a lot of like rules a lot of business rules a lot of roles so we have
00:36:31.160 this custom Ro DSL for our particular application which was built on top of up
00:36:36.680 K policy because again with the policy system of every application is different like what you need is not like oh this
00:36:42.920 is the exact solution this is the building blocks to build your own solution for the authorization so what I
00:36:49.319 have here in my startup particularly I have something which I call it roll stacking that means that for example the
00:36:55.640 D the the D the director has whatever the director can do whatever the
00:37:01.599 supervisor can do the operator the cashier so I have like this Branch stre so I can just write this DSL which is
00:37:09.400 like building cashier can view withraw that means the supervisor the the
00:37:15.040 director the operator can also do that so I can list all the actions and everything here kind of works and I have
00:37:23.280 basically have one this oneliner to basically Define like a map of all my permission of all my systems so every
00:37:30.960 time I need to adjust those is very simple and right now this was like two weeks ago when I was making this slides
00:37:36.280 this file had like 273 lines from those lines like 20 are like just comments
00:37:41.839 dsls and things like that but there were a lot of rules and imagine those 200 if those were like classes with like 20
00:37:48.640 lines of class names with a lot of and I just want to have like this simple map
00:37:53.960 and the other thing which was very key for me is having those methods especially this one so application
00:38:00.920 policy authorized I can call it from everywhere I don't need to think about a class name I don't need to think anything so in my application I have
00:38:09.200 those things which I enforc one is called require account uh require
00:38:14.480 account feature so this checks the current account it blows up if the account doesn't canot do calendar but
00:38:20.800 the key here is find record find record is uh price to find this record and if
00:38:29.400 and it checks if the current user can actually see this record like for
00:38:37.960 example so here I can give it like a option authorized method so um I have
00:38:44.200 this system here where if the user for example cannot find a
00:38:49.880 record this blows up with Kitty policy access Denine I can actually put this in
00:38:55.760 my error reporting so I know somebody tried to access something they shouldn't most probably I have a link somewhere
00:39:02.040 which is not with the can and the user collected by mistake and I have like a system on out toiz but this is how find
00:39:08.599 record actually looks like it's a very simple method which every record to my
00:39:14.079 system goes to this thing because every most of the time when you are trying to find something you start from somewhere
00:39:21.400 so find record does like okay find me get me the scope or like figure out it
00:39:26.599 then goes to the out right like okay now can this user do whatever what with this
00:39:33.160 record and then it was I also have this check here is if I had this required
00:39:40.520 feature calendar and checking if the account of the record actually has this
00:39:45.599 feature because I can draw a about of things and having this method actually was very awesome like I started my
00:39:51.800 project this part here didn't exist like this part didn't exist with the fin record and because didn't
00:39:58.640 exist uh when I needed to add oh I need to restrict certain actions because we
00:40:04.400 now have two plans I just changed these two lines and it didn't change anything in my other controllers I also here this
00:40:11.960 method actually has five five six more things like for example if your account is deactivated you shouldn't be able to
00:40:18.440 do almost anything so I also had this logic here so this find record is very useful when you build like a system to
00:40:25.880 have like a single place where all the loading of Records come from because
00:40:32.960 it's very useful for you to kind of have have the central place to check the authorization not to think about oh I'll
00:40:38.880 think it on a controller level just think it on like a loading level uh another thing which I have
00:40:44.839 found out useful in my project in particular I have noticed a lot of people also do it is a lot of Records in
00:40:51.280 our system are trees like they have an account the account owns something else like a project the project on a task and
00:40:58.680 from like every record you can actually go to the root of the tree in some way and you care about the current user
00:41:06.520 related to the to their account level like so I also have this thing which is like accounts get me the account of the
00:41:13.160 record and in this way I can say is this feature available for this record where
00:41:18.240 I go to the tree get the account and just say Okay this feure available so yeah that's the kitty
00:41:24.680 policy gem but the idea here is not the gem itself because again it's 20 lines uh but the idea here is if you build
00:41:31.400 your authentication system you do it in such a way that it's kind of encapsulated and it happens in your whole system it's not something that the
00:41:38.079 developer should think of uh
00:41:45.200 okay and yep I need to just come back so yeah so so far so good so much
00:41:53.760 content you have no idea how much I get from this presentation so uh service objects like this is like
00:42:01.560 something where there is a lot of debates a lot of discussions and for me honestly that's my service object no
00:42:09.400 external libraries rails doesn't need to solve that it's just I do this for SE subject I just have a module I do extend
00:42:16.680 self I have like a usually method call if I need multiple methods I just have multiple methods everything else is
00:42:22.760 private I don't care like for me that's like the most simple way to start and doing stuff and then again if you need
00:42:29.400 you add like I I I have thought oh maybe I have like an application service object or something like this but there
00:42:35.800 is so much stuff you don't need here so yeah personally what I do is just I have like service objects which are just
00:42:42.480 basically functions like a globally named functions and I use the method call because I can actually use the the
00:42:49.440 Ruby building uh use of calls in different places like for example is it
00:42:55.800 uh Lambda is it like proc is it like a model with a class. a CO it's the same
00:43:01.720 so I can do a lot of my service object compos them uh the the thing I do mostly
00:43:07.920 is form objects and I have my own gem mini form which is basically the same
00:43:13.800 thing as active model model it just has some nice goodies which I have seen a
00:43:19.640 lot of projects when they basically have an application form which is their own version of action model and they compile
00:43:28.240 like different parts what they need and I think that's something good that people should do uh one area here which
00:43:35.559 I personally also like is uh and I have noticed a lot of people still still have
00:43:41.559 it in the controllers they need to fil all the fields but often you need one form per One
00:43:47.720 controller and often I have this method that I call submit and subit underscore where it's like I actually guessed what
00:43:54.400 are the permit required for params so I never I just get the params to the form
00:44:01.680 I don't need to think about oh how is the form named what's in this form if I had a new field I have to go not only to
00:44:08.480 the form UI I need to go to the controller I need to list it there I need to change like five things on the pipeline I just usually I just say okay
00:44:16.160 just get me all the attributes names from this particular form and I care about that the form is my shield from
00:44:22.079 like all the fields and it works well most uh because
00:44:27.800 the the form is like a nice shoot it's not you are not calling the model directly plus I I I I change this
00:44:35.839 default option in rails every form has an S field and by default the form field
00:44:42.160 is the name of the model it's like let's say projects and it's name project let's
00:44:47.760 say it's account template it's account template and I have found this very unuseful for me because every time I
00:44:54.480 change the model name or I change the form I need to go to five places and change the params whatever thingy so all
00:45:01.920 my forms basically the form the input form is form because why not never seen
00:45:09.040 something it's very simple it works quite well and yeah and and the other thing
00:45:15.599 with the forms is I can just have my controllers mostly look like that I can
00:45:21.520 say okay this is a calendar event controller so I this controller so I can
00:45:27.319 just say Okay create me a form of the ATD with an event I can
00:45:33.280 do okay get me the calendar ATD form with like find event I can say a submit
00:45:39.280 the submit would get the params directly gets the params you don't have a separate method which lists all the
00:45:45.000 params with all the fields and again I have the find event here which uses the
00:45:50.079 F tracker to make sure the current user can actually manage this event and I use
00:45:56.079 this very old extracted from where yeah does that work with like more oh yeah
00:46:03.599 like my version the one I use in my application works with the like deep nestings and it validates the
00:46:08.880 keys I didn't show the whole code because it's like 30 lines and it's to early in the morning and this is non
00:46:14.240 alcoholic yet so it's uh but yeah it also I have I show I can show the
00:46:19.400 version with like the the B big deep nesting because I also have like delegations and things like that
00:46:25.079 connected with the thing but another another thing I use I think I'm the only one still using the gem responders from
00:46:31.160 like 200 years ago which was part of real then they remove it but I still really like this respond withd because
00:46:38.520 what respond withd does is it checks the record and if the error if the record
00:46:43.800 has validation errors it renders the error page if it doesn't have validation
00:46:49.040 errors it just goes to this location and it's like how many times do I need to write if
00:46:55.680 valid uh remember that so I just have this it's it's a bit of magical but I
00:47:01.559 still like it if this gem stop working I'll just in line it and use it myself
00:47:08.359 huh controller I if I need it I would do it like this is like really and also
00:47:13.760 this the other nice thing about this gem is it automatically also sets The Flash message and because it automatically in
00:47:20.319 my application particularly all the flash messages are connected to the internalization system so I actually
00:47:27.240 have flash message This Record was created and it's already translated and it all works and I don't need to think
00:47:32.680 about all the namings and when new developer starts working on my project they don't need to think about what's my
00:47:39.400 naming schedule for Flash events that that what would happen is they're going to implement this it's going to blow up
00:47:45.119 and tell them you don't have this key registered and they just would register the key and then they would
00:47:51.400 automatically do translations and it will be translated into five other languages and who cares
00:47:57.760 yeah but that's the thing what I do is I do I found out we talk about in the real Community a lot about the service
00:48:03.680 objects and not so much about the form objects and actually the form objects are I think the one separation between
00:48:10.240 like a service object and a form object is I think very important because the form object deals with like the user
00:48:15.920 input and it packages it for the user it has a lot which active model is really
00:48:21.559 great at and the service object is something that more okay doesn't have to
00:48:26.960 200 error States is just a function and operation and having this separation is
00:48:32.880 very useful uh like for example service objects often times are call from active jobs or like some external thing while
00:48:40.200 the form objects always work from like a user input also I have like a graph API
00:48:45.640 and the graph mutations are perfect for form objects I often in my previous company we had form objects for a UI
00:48:53.319 form in the admin and we used the same form object for for a graph mutation and they map very well to
00:49:00.480 that so yeah oh I'm getting to the end and it's almost two minutes left so
00:49:07.400 query object so often this is another thing often we need to build stuff like
00:49:13.400 that this is for my application when we have this very complex search search form we have some statistics for that
00:49:20.760 search and we have search results and this is a gem I created like 15 years
00:49:25.960 ago really literally 15 years ago it's called search object I'm very creative with names if you notice uh so search
00:49:34.200 object allows you to create like this form is basically search Finance transaction search filters params
00:49:41.520 results and usually if you need to implement an object like this I used to do like this 15 years ago I create this
00:49:48.960 class I get the filters and I basically do this if the filter is this just
00:49:54.480 reiterate to the scope and and get me like pagination like add me the pages
00:50:00.680 here and and again cat me the results because
00:50:06.359 again the results if I call it 20 times I just want one query so I started with this like 15 years ago I build this into
00:50:13.319 a gem where it's like I include the search object with kaminari for pagination I set the initial scope I get
00:50:21.400 the options and I have more again this is how we get the options it's basically the same same
00:50:27.960 thing and also this is actually dady added this 20 12 years ago you added
00:50:33.280 this feature the width yeah you added the width and the nice thing about this
00:50:38.440 having this object even if you don't use my gym if you just use the like you write this by yourself having an object
00:50:44.839 having a place to put stuff on you can add methods for oh give me the balance give me the income give me the expense
00:50:51.559 give me the count like this part you can just add it to your classes you don't need like Library you can just have like
00:50:58.079 a class that combines those searches and it can give you the statistics B based
00:51:03.520 on all those filters also if you have more advanced forms those forms have dropdowns and for example having this
00:51:10.280 class I can actually put in this class this is the cashier options this is the source options this is the type options
00:51:16.640 and having all of this in a central place in this search object is very useful to build it everywhere and yeah I
00:51:23.040 right now in my startup I use it a lot for UI forms the one I showed you and the nice thing is if you have this you
00:51:29.760 can actually build table in UI across of that because it's like a static right uh
00:51:34.920 it's like a standardized also I have used the same strategy with search object for apis like if you have like an
00:51:41.480 API or graph API you can have those composition of search objects so this is
00:51:47.440 another thing which is is very useful to think about like okay in a rilos project
00:51:52.599 this is all the things we have so yeah this is again in most ra project you
00:51:57.799 everybody has variations of those things and the big question is where do those things live I think this is another
00:52:05.119 thing so if you think about it what I told you today maybe in my app I have app policies services forms and queries
00:52:11.040 and that's great and that's great but I have other projects like I
00:52:18.920 have let's say I have the external stuff which I would name it app Network because this goes to the network oh I
00:52:26.200 have the validators as well maybe we have decorators maybe we have presenters because you know the big the big
00:52:31.920 difference between presenters decorator they shouldn't be mixed because people get confused uh we also have notifiers
00:52:37.799 for like notifications oh we have up values because maybe we have some value objects and we have heard it's a very
00:52:43.760 useful thing also maybe we need some support code like I have a library internal library for my project called
00:52:49.440 angry support for like the support Library also maybe ERS is also useful
00:52:55.760 because sometimes you don't know where to put stuff in that and again it's always a lot of discussions where to put
00:53:01.720 stuff and again for me it was always like oh my God and it's like oh we should put
00:53:08.040 everything in one folder like oh man this is such a mess then it's like okay put every object in their own folder it
00:53:15.119 becomes more messy and again I'm back to everything in one folder what I do in
00:53:20.480 all my projects right now is basically I have up lip so I move the lip from like down there to uplip because it's I think
00:53:27.480 very important and I use aggressive namespacing for the domain so for example I have search if you notied I
00:53:34.760 had the finance transaction search object I didn't have search transactions
00:53:39.960 I I start with the domain because it was very useful because when I go to my finance name space I see oh transaction
00:53:46.599 search transaction form transaction edit transaction edit whatever whatever now I
00:53:51.839 might maybe have transaction Finance transaction and I can group all those things so I have this and and this is
00:53:58.520 what right now how everything looks in my lip folder you you can notice my
00:54:03.839 domain I have the accounts I have the angry batch I have Angry support Apartments external export funds graph
00:54:12.359 heating which will be very useful right now if I look at that messaging payment
00:54:18.880 uh registration schedule search and all the other stuff but it's here by domain so when a developer starts thinking
00:54:25.839 about stuff they can do the fuzzy Search and say oh I search for the form object but it's Finance so I do Finance form and it
00:54:33.000 leaves me all the forms and yeah it's like a lot of like the YOLO kind of like everything is there everything is great
00:54:39.480 and everybody's so happy so yeah this is a lot of the stuff which again I was like okay every time I start a new rails
00:54:46.400 project I'm going to the slides and just copy pasting stuff because I don't do
00:54:51.520 rails project that often because it's so much work to copy paste all those settings but those I found like a lot of
00:54:58.599 like the stuff that's kind of like the missing parts and yeah that's pretty much all and I'm done on time
Explore all talks recorded at Ruby Banitsa Conf 2024