check.vim: Improve po message checks
Signed-off-by: Antonio Giovanni Colombo <azc100@gmail.com> Signed-off-by: Christian Brabandt <cb@256bit.org>
This commit is contained in:
		
				
					committed by
					
						 Christian Brabandt
						Christian Brabandt
					
				
			
			
				
	
			
			
			
						parent
						
							aa55185100
						
					
				
				
					commit
					2566a9f290
				
			
							
								
								
									
										114
									
								
								src/po/check.vim
									
									
									
									
									
								
							
							
						
						
									
										114
									
								
								src/po/check.vim
									
									
									
									
									
								
							| @ -1,16 +1,21 @@ | |||||||
| " Vim script for checking .po files. | " Vim script for checking .po files. | ||||||
| " | " | ||||||
| " Go through the file and verify that: | " Goes through the xx.po file (more than once) | ||||||
| " - All %...s items in "msgid" are identical to the ones in "msgstr". | " and verify various congruences | ||||||
| " - An error or warning code in "msgid" matches the one in "msgstr". | " See the comments in the code | ||||||
|  |  | ||||||
| " Last Update: 2025 Jul 10 | " Last Update: 2025 Jul 22 | ||||||
|  |  | ||||||
| if 1	" Only execute this if the eval feature is available. | if 1" Only execute this if the eval feature is available. | ||||||
|  |  | ||||||
| " using line continuation | " Using line continuation (set cpo to vim default value) | ||||||
|  | let s:save_cpo = &cpo | ||||||
| set cpo&vim | set cpo&vim | ||||||
|  |  | ||||||
|  | " This only works when 'wrapscan' is not set. | ||||||
|  | let s:save_wrapscan = &wrapscan | ||||||
|  | set nowrapscan | ||||||
|  |  | ||||||
| " Function to get a split line at the cursor. | " Function to get a split line at the cursor. | ||||||
| " Used for both msgid and msgstr lines. | " Used for both msgid and msgstr lines. | ||||||
| " Removes all text except % items and returns the result. | " Removes all text except % items and returns the result. | ||||||
| @ -43,9 +48,15 @@ func! GetMline() | |||||||
|   return substitute(idline, '[^%]*\(%([1-9][0-9]*\$)\=[-+ #''.0-9*]*l\=[dsuxXpoc%]\)\=', '\1', 'g') |   return substitute(idline, '[^%]*\(%([1-9][0-9]*\$)\=[-+ #''.0-9*]*l\=[dsuxXpoc%]\)\=', '\1', 'g') | ||||||
| endfunc | endfunc | ||||||
|  |  | ||||||
| " This only works when 'wrapscan' is not set. | func! CountNl(first, last) | ||||||
| let s:save_wrapscan = &wrapscan |   let nl = 0 | ||||||
| set nowrapscan |   for lnum in range(a:first, a:last) | ||||||
|  |     let nl += count(getline(lnum), "\n") | ||||||
|  |   endfor | ||||||
|  |   return nl | ||||||
|  | endfunc | ||||||
|  |  | ||||||
|  | " main | ||||||
|  |  | ||||||
| " Start at the first "msgid" line. | " Start at the first "msgid" line. | ||||||
| let wsv = winsaveview() | let wsv = winsaveview() | ||||||
| @ -57,19 +68,25 @@ keeppatterns /^msgid\> | |||||||
| let error = 0 | let error = 0 | ||||||
|  |  | ||||||
| while 1 | while 1 | ||||||
|  |   " for each "msgid" | ||||||
|  |  | ||||||
|  |   " check msgid "Text;editor;" | ||||||
|  |   " translation must have two ";" as well | ||||||
|   let lnum = line('.') |   let lnum = line('.') | ||||||
|   if getline(lnum) =~ 'msgid "Text;.*;"' |   if getline(lnum) =~ 'msgid "Text;.*;"' | ||||||
|     if getline(lnum + 1) !~ '^msgstr "\([^;]\+;\)\+"' |     if getline(lnum + 1) !~ '^msgstr "\([^;]\+;\)\+"$' | ||||||
|       echomsg 'Mismatching ; in line ' . (lnum + 1) |       echomsg 'Mismatching ; in line ' . (lnum + 1) | ||||||
|       echomsg 'Did you forget the trailing semicolon?' |       echomsg 'Wrong semicolon count' | ||||||
|       if error == 0 |       if error == 0 | ||||||
|         let error = lnum + 1 |         let error = lnum + 1 | ||||||
|       endif |       endif | ||||||
|     endif |     endif | ||||||
|   endif |   endif | ||||||
|  |  | ||||||
|  |   " check for equal number of % in msgid and msgstr | ||||||
|  |   " it is skipping the no-c-format strings | ||||||
|   if getline(line('.') - 1) !~ "no-c-format" |   if getline(line('.') - 1) !~ "no-c-format" | ||||||
|     " go over the "msgid" and "msgid_plural" lines |     " skip the "msgid_plural" lines | ||||||
|     let prevfromline = 'foobar' |     let prevfromline = 'foobar' | ||||||
|     let plural = 0 |     let plural = 0 | ||||||
|     while 1 |     while 1 | ||||||
| @ -80,7 +97,7 @@ while 1 | |||||||
|       if prevfromline != 'foobar' && prevfromline != fromline |       if prevfromline != 'foobar' && prevfromline != fromline | ||||||
|             \ && (plural != 1 |             \ && (plural != 1 | ||||||
|             \     || count(prevfromline, '%') + 1 != count(fromline, '%')) |             \     || count(prevfromline, '%') + 1 != count(fromline, '%')) | ||||||
| 	echomsg 'Mismatching % in line ' . (line('.') - 1) |         echomsg 'possibly mismatching % in line ' . (line('.') - 1) | ||||||
|         echomsg 'msgid: ' . prevfromline |         echomsg 'msgid: ' . prevfromline | ||||||
|         echomsg 'msgid: ' . fromline |         echomsg 'msgid: ' . fromline | ||||||
|         if error == 0 |         if error == 0 | ||||||
| @ -93,6 +110,7 @@ while 1 | |||||||
|       let prevfromline = fromline |       let prevfromline = fromline | ||||||
|     endwhile |     endwhile | ||||||
|  |  | ||||||
|  |     " checks that for each 'msgid' there is a 'msgstr' | ||||||
|     if getline('.') !~ '^msgstr' |     if getline('.') !~ '^msgstr' | ||||||
|       echomsg 'Missing "msgstr" in line ' . line('.') |       echomsg 'Missing "msgstr" in line ' . line('.') | ||||||
|       if error == 0 |       if error == 0 | ||||||
| @ -100,12 +118,14 @@ while 1 | |||||||
|       endif |       endif | ||||||
|     endif |     endif | ||||||
|  |  | ||||||
|     " check all the 'msgstr' lines |     " check all the 'msgstr' lines have the same number of '%' | ||||||
|  |     " only the number of '%' is checked, | ||||||
|  |     " %d vs. %s or %d vs. %ld  go undetected | ||||||
|     while getline('.') =~ '^msgstr' |     while getline('.') =~ '^msgstr' | ||||||
|       let toline = GetMline() |       let toline = GetMline() | ||||||
|       if fromline != toline |       if fromline != toline | ||||||
|             \ && (plural == 0 || count(fromline, '%') != count(toline, '%') + 1) |             \ && (plural == 0 || count(fromline, '%') != count(toline, '%') + 1) | ||||||
| 	echomsg 'Mismatching % in line ' . (line('.') - 1) |         echomsg 'possibly mismatching % in line ' . (line('.') - 1) | ||||||
|         echomsg 'msgid: ' . fromline |         echomsg 'msgid: ' . fromline | ||||||
|         echomsg 'msgstr: ' . toline |         echomsg 'msgstr: ' . toline | ||||||
|         if error == 0 |         if error == 0 | ||||||
| @ -142,14 +162,6 @@ if search('msgid "\("\n"\)\?\([EW][0-9]\+:\).*\nmsgstr "\("\n"\)\?[^"]\@=\2\@!') | |||||||
|   endif |   endif | ||||||
| endif | endif | ||||||
|  |  | ||||||
| func! CountNl(first, last) |  | ||||||
|   let nl = 0 |  | ||||||
|   for lnum in range(a:first, a:last) |  | ||||||
|     let nl += count(getline(lnum), "\n") |  | ||||||
|   endfor |  | ||||||
|   return nl |  | ||||||
| endfunc |  | ||||||
|  |  | ||||||
| " Check that the \n at the end of the msgid line is also present in the msgstr | " Check that the \n at the end of the msgid line is also present in the msgstr | ||||||
| " line.  Skip over the header. | " line.  Skip over the header. | ||||||
| 1 | 1 | ||||||
| @ -175,6 +187,55 @@ while 1 | |||||||
|   endif |   endif | ||||||
| endwhile | endwhile | ||||||
|  |  | ||||||
|  | " Check that the eventual continuation of 'msgstr' is well formed | ||||||
|  | " final '""', '\n"', ' "' are OK | ||||||
|  | " Beware, it can give false positives if the message is split | ||||||
|  | " in the middle of a word | ||||||
|  | 1 | ||||||
|  | keeppatterns /^"MIME-Version: | ||||||
|  | while 1 | ||||||
|  |   let lnum = search('^msgid\>') | ||||||
|  |   if lnum <= 0 | ||||||
|  |     break | ||||||
|  |   endif | ||||||
|  |   " "msgstr" goes from strlnum to end-1 | ||||||
|  |   let strlnum = search('^msgstr\>') | ||||||
|  |   let end = search('^$') | ||||||
|  |   if end <= 0 | ||||||
|  |     let end = line('$') + 1 | ||||||
|  |   endif | ||||||
|  |   " only if there is a continuation line... | ||||||
|  |   if end > strlnum + 1 | ||||||
|  |     let ilnum = strlnum | ||||||
|  |     while ilnum < end - 1 | ||||||
|  |       let iltype = 0 | ||||||
|  |       if getline( ilnum ) =~ "^msgid_plural" | ||||||
|  |         let iltype = 99 | ||||||
|  |       endif | ||||||
|  |       if getline( ilnum ) =~ "^msgstr[" | ||||||
|  |         let iltype = 98 | ||||||
|  |       endif | ||||||
|  |       if getline( ilnum ) =~ "\"\"" | ||||||
|  |         let iltype = 1 | ||||||
|  |       endif | ||||||
|  |       if getline( ilnum ) =~ " \"$" | ||||||
|  |         let iltype = 2 | ||||||
|  |       endif | ||||||
|  |       if getline( ilnum ) =~ "\\\\n\"$" | ||||||
|  |         let iltype = 3 | ||||||
|  |       endif | ||||||
|  |       if iltype == 0 | ||||||
|  |         echomsg 'Possibly incorrect final at line: ' . ilnum | ||||||
|  |         " TODO: make this an error | ||||||
|  |         " if error == 0 | ||||||
|  |         "   let error = ilnum | ||||||
|  |         " endif | ||||||
|  |       endif | ||||||
|  |       let ilnum += 1 | ||||||
|  |     endwhile | ||||||
|  |   endif | ||||||
|  | endwhile | ||||||
|  |  | ||||||
| " Check that the file is well formed according to msgfmts understanding | " Check that the file is well formed according to msgfmts understanding | ||||||
| if executable("msgfmt") | if executable("msgfmt") | ||||||
|   let filename = expand("%") |   let filename = expand("%") | ||||||
| @ -257,7 +318,14 @@ else | |||||||
|   exe error |   exe error | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | " restore original wrapscan | ||||||
| let &wrapscan = s:save_wrapscan | let &wrapscan = s:save_wrapscan | ||||||
| unlet s:save_wrapscan | unlet s:save_wrapscan | ||||||
|  |  | ||||||
|  | " restore original cpo | ||||||
|  | let &cpo = s:save_cpo | ||||||
|  | unlet s:save_cpo | ||||||
|  |  | ||||||
| endif | endif | ||||||
|  |  | ||||||
|  | " vim:sts=2:sw=2:et | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user