tdf#115863: fix TIFF import
[LibreOffice.git] / .git-hooks / pre-commit
blob5ab5394668279e47c61c8ba86e7128581e7d8520
1 #!/usr/bin/env perl
3 # A hook script to verify what is about to be committed.
4 # Called by "git commit" with no arguments. The hook should
5 # exit with non-zero status after issuing an appropriate message
6 # if it wants to stop the commit.
8 use strict;
9 use lib "solenv/clang-format";
10 #use File::Copy;
11 #use Cwd;
13 $ENV{LC_ALL} = "C";
15 sub check_whitespaces($)
17 my ($h) = @_;
18 my $src_limited = "c|cpp|cxx|h|hrc|hxx|idl|inl|java|swift|map|MK|pmk|pl|pm|sdi|sh|src|tab|ui|xcu|xml|xsl";
19 my $src_full = "c|cpp|cxx|h|hrc|hxx|idl|inl|java|swift|map|mk|MK|pmk|pl|pm|sdi|sh|src|tab|ui|xcu|xml|xsl";
21 my $found_bad = 0;
22 my $filename;
23 my $reported_filename = "";
24 my $lineno;
25 sub bad_line
27 my ($why, $line, $file_filter) = @_;
28 if (!defined $file_filter || $filename =~ /\.($file_filter)$/)
30 if (!$found_bad)
32 print STDERR "*\n";
33 print STDERR "* You have some suspicious patch lines:\n";
34 print STDERR "*\n";
35 $found_bad = 1;
37 if ($reported_filename ne $filename)
39 print STDERR "* In $filename\n";
40 $reported_filename = $filename;
42 print STDERR "* $why (line $lineno)\n";
43 print STDERR "$filename:$lineno:$line\n";
46 open( FILES, "git-diff-index -p -M --cached $h |" ) || die "Cannot run git diff-index.";
47 while (<FILES>)
49 if (m|^diff --git a/(.*) b/\1$|)
51 $filename = $1;
52 next;
54 if (/^@@ -\S+ \+(\d+)/)
56 $lineno = $1 - 1;
57 next;
59 if (/^ /)
61 $lineno++;
62 next;
64 if (s/^\+//)
66 $lineno++;
67 chomp;
68 if (/\s$/)
70 bad_line("trailing whitespace", $_ , $src_limited);
72 if (/\r$/)
74 bad_line("DOS lineends", $_ , $src_limited);
76 if (/\s* /)
78 bad_line("indent with Tab", $_, $src_limited);
80 if (/^(?:[<>=]){7}$/)
82 bad_line("unresolved merge conflict", $src_full);
84 if (/SAL_DEBUG/)
86 bad_line("temporary debug in commit", $_, $src_limited);
88 if (/<property name="use_markup">True<\/property>/)
90 bad_line("use font attributes instead of use-markup", $_, "ui");
92 if (/<property name="tooltip_markup"/ )
94 bad_line("use tooltip_text instead of tooltip_markup", $_, "ui");
96 if ((/translatable="yes"/) and not(/context=/))
98 bad_line("translatable .ui file line without context", $_, "ui");
100 if ((/<interface/) and not(/domain=/))
102 bad_line(".ui file without translation domain", $_, "ui");
106 if ( $found_bad)
108 exit($found_bad);
112 sub check_author()
114 my $author = `git var GIT_AUTHOR_IDENT`;
115 chomp $author;
116 if ($author =~ /^Your Name <you\@example.com>/)
118 print("ERROR: You have a suspicious author identity: '$author'\n");
119 exit(1);
123 sub check_style($)
125 if (! -e "solenv/clang-format/ClangFormat.pm")
127 # Commit happens in a submodule.
128 return;
131 require ClangFormat;
132 ClangFormat->import();
134 my ($h) = @_;
135 my $src = ClangFormat::get_extension_regex();
136 my @bad_names = ();
137 my $blacklist_names = ClangFormat::get_blacklist();
138 my $clang_format = ClangFormat::find();
140 # Get a list of non-deleted changed files.
141 open (FILES, "git diff-index --cached --diff-filter=AM --name-only $h |") || die "Cannot run git diff.";
142 while (my $filename = <FILES>)
144 chomp $filename;
145 if ($filename =~ /\.($src)$/ and !exists($blacklist_names->{$filename}))
147 if (!defined($clang_format))
149 my $version = ClangFormat::get_wanted_version();
150 my $opt_lo = ClangFormat::get_own_directory();
152 print("\nWARNING: Commit touches new (non-blacklisted) files, but no clang-format"
153 . " ${version}\n");
154 print(" found (via CLANG_FORMAT or PATH env vars, or in ${opt_lo}).\n\n");
156 my $platform = "linux64";
157 my $download = "wget";
158 if ($^O eq "cygwin")
160 $platform = "win.exe";
162 elsif ($^O eq "darwin")
164 $platform = "mac";
165 $download = "curl -O";
168 print("To get a suitable binary, please do:\n\n");
169 print("mkdir -p $opt_lo\n");
170 print("cd $opt_lo\n");
171 print("$download https://dev-www.libreoffice.org/bin/clang-format-$version-$platform\n");
172 print("cp clang-format-$version-$platform clang-format\n");
173 print("chmod +x clang-format\n\n");
175 print("(Or read solenv/clang-format/README on how to build it yourself.\n");
176 return;
178 if (!ClangFormat::check_style($clang_format, $filename))
180 push @bad_names, $filename;
185 # Enforce style.
186 if (scalar @bad_names)
188 my $autostyle = `git config libreoffice.autostyle`;
189 chomp $autostyle;
190 if ($autostyle ne "true")
192 print("\nThe above differences were found between the code to commit \n");
193 print("and the clang-format rules. You can apply these changes with:\n");
194 print("\n$clang_format -i " . join(" ", @bad_names) . "\n\n");
195 print("Aborting commit. Apply changes and commit again or skip checking\n");
196 print("with --no-verify (not recommended).\n");
197 exit(1);
199 else
201 # 'git config libreoffice.autostyle true' was invoked to run
202 # clang-format automatically.
203 print("\nThe above differences were found between the code to commit \n");
204 print("and the clang-format rules. Fixing these now automatically.\n");
205 print("Running '$clang_format -i " . join(" ", @bad_names) . "' for you...\n");
206 system("$clang_format -i " . join(" ", @bad_names));
207 # TODO this stages all local modifications, staging originally
208 # unstaged hunks.
209 system("git add " . join(" ", @bad_names));
210 print("Done.\n");
215 # Do the work :-)
217 # Initial commit: diff against an empty tree object
218 my $against="4b825dc642cb6eb9a060e54bf8d69288fbee4904";
219 if ( system( "git rev-parse --verify HEAD >/dev/null 2>&1" ) == 0 )
221 $against="HEAD"
224 # If you want to allow non-ascii filenames set this variable to true.
225 my $allownonascii=`git config hooks.allownonascii`;
227 # Cross platform projects tend to avoid non-ascii filenames; prevent
228 # them from being added to the repository. We exploit the fact that the
229 # printable range starts at the space character and ends with tilde.
230 if ( $allownonascii ne "true" &&
231 # Note that the use of brackets around a tr range is ok here, (it's
232 # even required, for portability to Solaris 10's /usr/bin/tr), since
233 # the square bracket bytes happen to fall in the designated range.
234 `git diff --cached --name-only --diff-filter=A -z $against | \
235 LC_ALL=C tr -d '[ -~]\\0'` ne "" )
237 print <<EOM;
238 Error: Attempt to add a non-ascii file name.
240 This can cause problems if you want to work
241 with people on other platforms.
243 To be portable it is advisable to rename the file ...
245 If you know what you are doing you can disable this
246 check using:
248 git config hooks.allownonascii true
251 exit( 1 );
254 # Block large files.
255 open( FILES, "git diff --cached --name-only --diff-filter=A -z $against |" ) || die "Cannot run git diff-index.";
256 while (<FILES>)
258 if (/\.ui$/) # .ui files can get large
260 continue;
262 if (/\.xsl$/) # XSLT
264 continue;
266 my $size = `git cat-file -s :$_`;
267 # For now let's say large is 500KB
268 my $limit = 500;
269 if ($size > $limit * 1024)
271 print "Error: Attempt to add a large file: $_, pleasy try to fit into $limit KB.\n";
272 exit( 1 );
276 # fix whitespace in code
277 check_whitespaces( $against);
279 # fix style in code
280 check_style($against);
282 # catch missing author info
283 check_author();
285 # all OK
286 exit( 0 );
287 # vi:set shiftwidth=4 expandtab: