Skip to content

Commit

Permalink
Add Git Commit hook for code checks (#21)
Browse files Browse the repository at this point in the history
* Add pre-commit command

* Add step to sort xcode

* Sort xcode project file

* Add note in README
  • Loading branch information
jocmp committed Feb 29, 2024
1 parent 5627c41 commit b0ce7ba
Show file tree
Hide file tree
Showing 7 changed files with 363 additions and 134 deletions.
14 changes: 14 additions & 0 deletions .pre-commit-config.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
repos:
- repo: https://github.com/pre-commit/pre-commit-hooks
rev: v2.3.0
hooks:
- id: check-yaml
- id: end-of-file-fixer
- id: trailing-whitespace
- repo: local
hooks:
- id: sort-xcode-project-file
name: Sort Xcode Project File
entry: ./scripts/sort-xcode-project-file.sh
language: script
files: .*\.pbxproj
2 changes: 1 addition & 1 deletion .python-version
Original file line number Diff line number Diff line change
@@ -1 +1 @@
3.9.7
3.12.2
262 changes: 131 additions & 131 deletions ArtAtGVSU.xcodeproj/project.pbxproj

Large diffs are not rendered by default.

9 changes: 7 additions & 2 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,21 @@ SHELL:=/usr/bin/env bash

FASTLANE ?= bundle exec fastlane
SECRETS_DIR ?= ../art-at-gvsu-secrets/ios
PIP ?= pip3

secrets:
cp -v $(SECRETS_DIR)/ArtAtGVSU/GoogleService-Info.plist ./ArtAtGVSU/GoogleService-Info.plist
cp -v $(SECRETS_DIR)/ArtAtGVSU/Secrets.plist ./ArtAtGVSU/Secrets.plist

.PHONY: ci-test ci-secrets deploy-beta deploy-production deps secrets
.PHONY: ci-test ci-secrets deploy-beta deploy-production deps secrets bootstrap

deps:
bundle install
pip3 install bumpver==2023.1125
$(PIP) install bumpver==2023.1125
$(PIP) install pre-commit==3.6.2

bootstrap: deps
pre-commit install

.PHONY: bump-version
bump-version:
Expand Down
11 changes: 11 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,17 @@
[img_version]: https://img.shields.io/static/v1.svg?label=CalVer&message=2023.12.1012&color=blue
[url_version]: https://github.com/gvsucis/art-at-gvsu-v3

## Committing

Run the following command before adding a new commit

```
make bootstrap
```

This will install several Git commit hooks via [pre-commit](https://pre-commit.com/) to ensure
the Xcode project file is sorted among other code quality checks.

## Release

### Testflight
Expand Down
196 changes: 196 additions & 0 deletions scripts/sort-xcode-project-file.pl
Original file line number Diff line number Diff line change
@@ -0,0 +1,196 @@
#!/usr/bin/env perl

# Copyright (C) 2007-2021 Apple Inc. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
#
# 1. Redistributions of source code must retain the above copyright
# notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
# notice, this list of conditions and the following disclaimer in the
# documentation and/or other materials provided with the distribution.
# 3. Neither the name of Apple Inc. ("Apple") nor the names of
# its contributors may be used to endorse or promote products derived
# from this software without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

# Script to sort "children" and "files" sections in Xcode project.pbxproj files

use strict;
use warnings;

use File::Basename;
use File::Spec;
use File::Temp qw(tempfile);
use Getopt::Long;

sub sortChildrenByFileName($$);
sub sortFilesByFileName($$);

# Some files without extensions, so they can sort with the other files.
# Otherwise, names without extensions are assumed to be groups or directories and sorted last.
my %isFile = map { $_ => 1 } qw(
create_hash_table
);

my $printWarnings = 1;
my $showHelp;

my $getOptionsResult = GetOptions(
'h|help' => \$showHelp,
'w|warnings!' => \$printWarnings,
);

if (scalar(@ARGV) == 0 && !$showHelp) {
print STDERR "ERROR: No Xcode project files (project.pbxproj) listed on command-line.\n";
undef $getOptionsResult;
}

if (!$getOptionsResult || $showHelp) {
print STDERR <<__END__;
Usage: @{[ basename($0) ]} [options] path/to/project.pbxproj [path/to/project.pbxproj ...]
-h|--help show this help message
-w|--[no-]warnings show or suppress warnings (default: show warnings)
__END__
exit 1;
}

for my $projectFile (@ARGV) {
if (basename($projectFile) =~ /\.xcodeproj$/) {
$projectFile = File::Spec->catfile($projectFile, "project.pbxproj");
}

if (basename($projectFile) ne "project.pbxproj") {
print STDERR "WARNING: Not an Xcode project file: $projectFile\n" if $printWarnings;
next;
}

# Grab the mainGroup for the project file.
my $mainGroup = "";
open(IN, "< $projectFile") || die "Could not open $projectFile: $!";
while (my $line = <IN>) {
$mainGroup = $2 if $line =~ m#^(\s*)mainGroup = ([0-9A-F]{24}( /\* .+ \*/)?);$#;
}
close(IN);

# Guess the basename of any umbrella header, based on the project name.
# Umbrella headers are sorted to the top of their Headers phase to work
# around rdar://104432605.
(my $umbrellaHeaderBasename = $projectFile) =~ s/.*\/(\w+?)(Legacy)?\.xcodeproj\/project\.pbxproj$/$1/;

my ($OUT, $tempFileName) = tempfile(
basename($projectFile) . "-XXXXXXXX",
DIR => dirname($projectFile),
UNLINK => 0,
);

# Clean up temp file in case of die()
$SIG{__DIE__} = sub {
close(IN);
close($OUT);
unlink($tempFileName);
};

my @lastTwo = ();
open(IN, "< $projectFile") || die "Could not open $projectFile: $!";
while (my $line = <IN>) {
if ($line =~ /^(\s*)files = \(\s*$/) {
print $OUT $line;
my $endMarker = $1 . ");";
my @files;
while (my $fileLine = <IN>) {
if ($fileLine =~ /^\Q$endMarker\E\s*$/) {
$endMarker = $fileLine;
last;
}
if ($fileLine =~ /$umbrellaHeaderBasename(Private)?\.h/) {
# Sort umbrella headers to the top. Needed until
# rdar://104432605 is fixed in all shipping Xcodes.
print $OUT $fileLine;
} else {
push @files, $fileLine;
}
}
print $OUT sort sortFilesByFileName @files;
print $OUT $endMarker;
} elsif ($line =~ /^(\s*)children = \(\s*$/) {
print $OUT $line;
my $endMarker = $1 . ");";
my @children;
while (my $childLine = <IN>) {
if ($childLine =~ /^\Q$endMarker\E\s*$/) {
$endMarker = $childLine;
last;
}
push @children, $childLine;
}
if ($lastTwo[0] =~ m#^\s+\Q$mainGroup\E = \{$#) {
# Don't sort mainGroup
print $OUT @children;
} elsif ($lastTwo[0] =~ m#\Q/* Products */\E = \{$#) {
# Don't sort Products
print $OUT @children;
} else {
print $OUT sort sortChildrenByFileName @children;
}
print $OUT $endMarker;
} else {
print $OUT $line;
}

push @lastTwo, $line;
shift @lastTwo if scalar(@lastTwo) > 2;
}
close(IN);
close($OUT);

unlink($projectFile) || die "Could not delete $projectFile: $!";
rename($tempFileName, $projectFile) || die "Could not rename $tempFileName to $projectFile: $!";
}

exit 0;

sub sortChildrenByFileName($$)
{
my ($a, $b) = @_;
my $aFileName = $1 if $a =~ /^\s*[A-Z0-9]{24} \/\* (.+) \*\/,$/;
my $bFileName = $1 if $b =~ /^\s*[A-Z0-9]{24} \/\* (.+) \*\/,$/;
my $aSuffix = $1 if $aFileName =~ m/\.([^.]+)$/;
my $bSuffix = $1 if $bFileName =~ m/\.([^.]+)$/;
my $aIsDirectory = !$aSuffix && !$isFile{$aFileName};
my $bIsDirectory = !$bSuffix && !$isFile{$bFileName};
return $bIsDirectory <=> $aIsDirectory if $aIsDirectory != $bIsDirectory;
if ($aFileName =~ /^UnifiedSource\d+/ && $bFileName =~ /^UnifiedSource\d+/) {
my $aNumber = $1 if $aFileName =~ /^UnifiedSource(\d+)/;
my $bNumber = $1 if $bFileName =~ /^UnifiedSource(\d+)/;
return $aNumber <=> $bNumber if $aNumber != $bNumber;
}
return lc($aFileName) cmp lc($bFileName) if lc($aFileName) ne lc($bFileName);
return $aFileName cmp $bFileName;
}

sub sortFilesByFileName($$)
{
my ($a, $b) = @_;
my $aFileName = $1 if $a =~ /^\s*[A-Z0-9]{24} \/\* (.+) in /;
my $bFileName = $1 if $b =~ /^\s*[A-Z0-9]{24} \/\* (.+) in /;
if ($aFileName =~ /^UnifiedSource\d+/ && $bFileName =~ /^UnifiedSource\d+/) {
my $aNumber = $1 if $aFileName =~ /^UnifiedSource(\d+)/;
my $bNumber = $1 if $bFileName =~ /^UnifiedSource(\d+)/;
return $aNumber <=> $bNumber if $aNumber != $bNumber;
}
return lc($aFileName) cmp lc($bFileName) if lc($aFileName) ne lc($bFileName);
return $aFileName cmp $bFileName;
}
3 changes: 3 additions & 0 deletions scripts/sort-xcode-project-file.sh
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
#!/usr/bin/env bash

perl ./scripts/sort-xcode-project-file.pl ./ArtAtGVSU.xcodeproj

0 comments on commit b0ce7ba

Please sign in to comment.