====== Modern Perl ====== Monday July 1, 2024 Last year, I wrote an article detailing [[posts:2023:2023.04.26_perl_really_that_bad|my experience re-creating a Python script in Perl]]. This was a pretty cumbersome exercise, but I didn’t use many of Perl’s newer features. I decided to re-visit this, leveraging some of these newer features to see how well it improves the process. ===== Enabling Features ===== To take advantage of specific features, use the aptly named ''feature'' pragma. You can enable an individual feature by name, and multiple features at once with ''qw()''. For example, this: use feature 'fc'; use feature 'say'; Is the same as this: use feature qw(fc say); Full example: use feature qw(fc say); $x = 'Test'; say "The case-folded version of $x is: " . fc $x; Output: The case-folded version of Test is: test Without ''use feature'', you’d get this: String found where operator expected (Do you need to predeclare "say"?) at ./test.pl line 7, near "say "The case-folded version of $x is: "" syntax error at ./test.pl line 7, near "say "The case-folded version of $x is: "" Execution of ./test.pl aborted due to compilation errors. You can also take advantage of **feature bundles**, enabling all of the features available as of a given version: # implicitly loads 5.36 feature bundle use v5.36; But, you must also have (at least) that version of Perl installed, of course: use v5.40; Perl v5.40.0 required--this is only v5.38.2, stopped at ./test.pl line 3. BEGIN failed--compilation aborted at ./test.pl line 3. ===== Useful Features ===== [[https://en.wikipedia.org/wiki/Perl_5_version_history|Lots of new features have been added in recent years]], but I’ll take advantage of only a small subset of them for this exercise. I’ll be using features available in 5.38.2 and earlier. ^ Feature ^ Version ^ Notes ^ | New ''class'' feature | 5.38.0 | Adds support for object-oriented code. | | New ''say'' built-in | 5.10.0 | Works like ''print'', adds a newline. | | Subroutine signatures no longer considered experimental | 5.36.0 | Supports named parameters in subroutines. | ===== Refactoring As Structured Code ===== We’ll start with the first subroutine that’s called. Here’s the original: sub exec_backup_set { my @source_paths = @{ $_[0] }; my $target_path = $_[1]; foreach my $source_path (@source_paths) { exec_backup( $source_path, $target_path ); } } And here’s the new version, using the ''signatures'' feature to implement subroutine arguments as lexical variables: sub exec_backup_set ( $target_path, @source_paths ) { foreach my $source_path (@source_paths) { exec_backup( $source_path, $target_path ); } } You’ll notice that using named arguments in the method signature is a lot cleaner. But, we have to swap the **source_paths** and **target_path** arguments. Why? Since **source_paths** is an array, it’s a ‘slurpy’ argument, meaning it will reference all remaining arguments in the signature. So, if it were the first argument, it would also absorb the contents of **target_path**. > A “slurpy” parameter is a list or hash parameter that “slurps up” all remaining arguments. Since any following parameters can’t receive values, there can be only one slurpy parameter. > > Slurpy parameters must come at the end of the signature and they must be positional. > > Slurpy parameters are optional by default. > > https://metacpan.org/pod/Method::Signatures#Slurpy-parameters Next, we’ll refactor the ''exec_backup'' subroutine. Here’s the original: sub exec_backup { my ( $source, $target ) = @_; my $proc_name = "rsync -lrtv --delete \"${source}\" \"${target}\""; unless ( -d $target ) { mkdir($target); } if ( -d $target ) { print("Syncing ${source}...\n"); system($proc_name); print("Synced ${source}\n"); } print("----------\n"); } For our updated version, we’ll use the ''signatures'' feature again, along with the ''say'' feature: sub exec_backup_modern ( $source, $target ) { my $proc_name = "rsync -lrtv --delete \"${source}\" \"${target}\""; unless ( -d $target ) { mkdir($target); } if ( -d $target ) { say("Syncing ${source}..."); system($proc_name); say("Synced ${source}"); } say("----------"); } Now we have enough information to refactor our entire script. Here’s the “modern” version of our [[posts:2023:2023.04.26_perl_really_that_bad|old script]]: use strict; use warnings; use feature 'signatures'; use feature 'say'; sub exec_backup ( $source, $target ) { my $proc_name = "rsync -lrtv --delete \"${source}\" \"${target}\""; unless ( -d $target ) { mkdir($target); } if ( -d $target ) { say("Syncing ${source}..."); system($proc_name); say("Synced ${source}"); } say("----------"); } sub exec_backup_set ( $target_path, @source_paths ) { foreach my $source_path (@source_paths) { exec_backup( $source_path, $target_path ); } } my @target_paths = ( "/target1/", "/target2/" ); # regular file sets my @regular_files = ( "/home/jimc/source1", "/home/jimc/source2", "/home/jimc/source3", "/home/jimc/source4" ); exec_backup_set( $target_paths[0], @regular_files ); # large file sets my @large_files = ( "/home/jimc/large1", "/home/jimc/large2" ); exec_backup_set( $target_paths[1], @large_files ); sleep(2); # pause before exiting With these changes, we have a script that’s more concise and readable. ===== Refactoring As Object-Oriented Code ===== We’ve improved the code, but what if we want to implement it using an object-oriented paradigm instead of structured? We can accomplish this using the ''class'' feature. Let’s start with a very simple example: use strict; use warnings; use feature 'class'; use feature 'say'; class MyUtil::Backup { method say_hello { say("Hello!"); } } my $backup_obj = MyUtil::Backup->new(); $backup_obj->say_hello; If you’re familiar with “old” Perl, this will look //very// unusual. We have a couple of new keywords we’re using, ''class'' and ''method''. In this example, we’ve defined a new class ''Backup'' in a namespace ''MyUtil''. Inside the class, we have a single callable method named ''say_hello''. We create an instance of the class named ''$backup_obj'' and then call the ''say_hello'' method. We run the code and… class is experimental at ./fullsync_oop.pl line 9. method is experimental at ./fullsync_oop.pl line 11. Hello! The code runs, but the interpreter is warning us about the experimental nature of the ''class'' and ''method'' keywords. We can suppress this with a ''no warnings'' directive. With our code now looking like this: use strict; use warnings; use feature 'class'; use feature 'say'; no warnings 'experimental::class'; class MyUtil::Backup { method say_hello { say("Hello!"); } } my $backup_obj = MyUtil::Backup->new(); $backup_obj->say_hello; We see this: Hello! Now let’s add a field to hold our name, and use it in our greeting: use strict; use warnings; use feature 'class'; use feature 'say'; no warnings 'experimental::class'; class MyUtil::Backup { field $name : param; method say_hello { say("Hello, $name!"); } } my $backup_obj = MyUtil::Backup->new( name => 'Jim' ); $backup_obj->say_hello; We’ve added a field named ''$name''. ''param'' indicates that the field value will be initialized in the constructor, which we call as ''MyUtil::Backup->new( name => 'Jim' )''. When we run, we see this: Hello, Jim! Now we’ve learned enough to refactor our structured script, and make it object-oriented: use strict; use warnings; use feature 'class'; use feature 'say'; no warnings 'experimental::class'; class MyUtil::Backup { method exec_backup ( $source, $target ) { my $proc_name = "rsync -lrtv --delete \"${source}\" \"${target}\""; unless ( -d $target ) { mkdir($target); } if ( -d $target ) { say("Syncing ${source}..."); system($proc_name); say("Synced ${source}"); } say("----------"); } method exec_backup_set ( $target_path, @source_paths ) { foreach my $source_path (@source_paths) { $self->exec_backup( $source_path, $target_path ); } } } my $backup_obj = MyUtil::Backup->new(); my @target_paths = ( "/target1/", "/target2/" ); # regular file sets my @regular_files = ( "/home/jimc/source1", "/home/jimc/source2", "/home/jimc/source3", "/home/jimc/source4" ); $backup_obj->exec_backup_set( $target_paths[0], @regular_files ); # large file sets my @large_files = ( "/home/jimc/large1", "/home/jimc/large2" ); $backup_obj->exec_backup_set( $target_paths[1], @large_files ); sleep(2); # pause before exiting This one’s a bit more verbose, but still pretty clean and readable. Note the syntax of the call to the ''exec_backup'' method from the ''exec_backup_set'' method: I had to prefix it with ''$self'' to indicate that a method of the current instance of the class is being called. ===== The Verdict ===== I still don’t consider Perl to be the friendliest language in the world, but taking advantage of modern features definitely improves the experience! ===== Links To References ===== https://perldoc.perl.org/feature https://perldoc.perl.org/5.38.0/perlclass