7017
|
1 ## Copyright (C) 2006, 2007 Bill Denney |
6045
|
2 ## |
6440
|
3 ## This file is part of Octave. |
6045
|
4 ## |
6440
|
5 ## Octave is free software; you can redistribute it and/or modify it |
|
6 ## under the terms of the GNU General Public License as published by |
7016
|
7 ## the Free Software Foundation; either version 3 of the License, or (at |
|
8 ## your option) any later version. |
6440
|
9 ## |
|
10 ## Octave is distributed in the hope that it will be useful, but |
|
11 ## WITHOUT ANY WARRANTY; without even the implied warranty of |
|
12 ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
13 ## General Public License for more details. |
6045
|
14 ## |
|
15 ## You should have received a copy of the GNU General Public License |
7016
|
16 ## along with Octave; see the file COPYING. If not, see |
|
17 ## <http://www.gnu.org/licenses/>. |
6045
|
18 |
|
19 ## -*- texinfo -*- |
6248
|
20 ## @deftypefn {Function File} {} compare_versions (@var{v1}, @var{v2}, @var{operator}) |
6045
|
21 ## Compares to version strings using the given @var{operator}. |
|
22 ## |
|
23 ## This function assumes that versions @var{v1} and @var{v2} are |
|
24 ## arbitrarily long strings made of numeric and period characters |
|
25 ## possibly followed by an arbitrary string (e.g. "1.2.3", "0.3", |
|
26 ## "0.1.2+", or "1.2.3.4-test1"). |
|
27 ## |
|
28 ## The version is first split into the numeric and the character parts |
|
29 ## then the parts are padded to be the same length (i.e. "1.1" would be |
|
30 ## padded to be like "1.1.0" when being compared with "1.1.1", and |
|
31 ## separately, the character parts of the strings are padded with |
|
32 ## nulls). |
|
33 ## |
|
34 ## The operator can be any logical operator from the set |
|
35 ## |
|
36 ## @itemize @bullet |
|
37 ## @item |
|
38 ## "==" |
|
39 ## equal |
|
40 ## @item |
|
41 ## "<" |
|
42 ## less than |
|
43 ## @item |
|
44 ## "<=" |
|
45 ## less than or equal to |
|
46 ## @item |
|
47 ## ">" |
|
48 ## greater than |
|
49 ## @item |
|
50 ## ">=" |
|
51 ## greater than or equal to |
|
52 ## @item |
|
53 ## "!=" |
|
54 ## not equal |
|
55 ## @item |
|
56 ## "~=" |
|
57 ## not equal |
|
58 ## @end itemize |
|
59 ## |
|
60 ## Note that version "1.1-test2" would compare as greater than |
|
61 ## "1.1-test10". Also, since the numeric part is compared first, "a" |
|
62 ## compares less than "1a" because the second string starts with a |
|
63 ## numeric part even though double("a") is greater than double("1"). |
|
64 ## @end deftypefn |
|
65 |
7017
|
66 ## Author: Bill Denney <denney@seas.upenn.edu> |
|
67 |
6045
|
68 ## TODO?: This allows a single equal sign "=" to indicate equality, do |
|
69 ## we want to require a double equal since that is the boolean operator? |
|
70 |
7069
|
71 function out = compare_versions (v1, v2, operator) |
6045
|
72 |
7125
|
73 if (nargin != 3) |
|
74 print_usage (); |
|
75 endif |
|
76 |
7069
|
77 ## Make sure that the version numbers are valid. |
|
78 if (! (ischar (v1) && ischar (v2))) |
|
79 error ("compare_versions: both version numbers must be strings"); |
|
80 elseif (size (v1, 1) != 1 || size (v2, 1) != 1) |
|
81 error ("compare_versions: version numbers must be a single row") |
6045
|
82 endif |
|
83 |
|
84 ## check and make sure that the operator is valid |
7069
|
85 if (! ischar (operator)) |
|
86 error ("compare_versions: operator must be a character string"); |
6045
|
87 elseif (numel (operator) > 2) |
7069
|
88 error("compare_versions: operator cannot be more than 2 characters long"); |
6045
|
89 endif |
|
90 |
|
91 ## trim off any character data that is not part of a normal version |
|
92 ## number |
|
93 numbers = "0123456789."; |
7069
|
94 |
|
95 v1firstchar = find (! ismember (v1, numbers), 1); |
|
96 v2firstchar = find (! ismember (v2, numbers), 1); |
|
97 if (! isempty (v1firstchar)) |
6045
|
98 v1c = v1(v1firstchar:length(v1)); |
|
99 v1nochar = v1(1:v1firstchar-1); |
|
100 else |
|
101 v1c = ""; |
|
102 v1nochar = v1; |
|
103 endif |
7069
|
104 if (! isempty (v2firstchar)) |
6045
|
105 v2c = v2(v2firstchar:length(v2)); |
|
106 v2nochar = v2(1:v2firstchar-1); |
|
107 else |
|
108 v2c = ""; |
|
109 v2nochar = v2; |
|
110 endif |
|
111 |
7069
|
112 v1n = str2num (split (v1nochar, ".")); |
|
113 v2n = str2num (split (v2nochar, ".")); |
6045
|
114 if ((isempty (v1n) && isempty (v1c)) || (isempty (v2n) && isempty(v2c))) |
7069
|
115 error ("compare_versions: given version strings are not valid: %s %s", |
|
116 v1, v2); |
6045
|
117 endif |
|
118 |
7069
|
119 ## Assume that any additional elements would be 0 if one is longer |
|
120 ## than the other. |
6045
|
121 maxnumlen = max ([length(v1n) length(v2n)]); |
|
122 if (length (v1n) < maxnumlen) |
|
123 v1n(length(v1n)+1:maxnumlen) = 0; |
|
124 endif |
|
125 if (length (v2n) < maxnumlen) |
|
126 v2n(length(v2n)+1:maxnumlen) = 0; |
|
127 endif |
|
128 |
7069
|
129 ## Assume that any additional character elements would be 0 if one is |
|
130 ## longer than the other. |
|
131 maxcharlen = max ([length(v1c), length(v2c)]); |
6045
|
132 if (length (v1c) < maxcharlen) |
|
133 v1c(length(v1c)+1:maxcharlen) = "\0"; |
|
134 endif |
|
135 if (length (v2c) < maxcharlen) |
|
136 v2c(length(v2c)+1:maxcharlen) = "\0"; |
|
137 endif |
|
138 |
7069
|
139 ## Determine the operator. |
6045
|
140 if any (ismember (operator, "=")) |
|
141 equal_op = true; |
|
142 else |
|
143 equal_op = false; |
|
144 end |
|
145 if any (ismember (operator, "~!")) |
|
146 not_op = true; |
|
147 else |
|
148 not_op = false; |
|
149 endif |
|
150 if any (ismember (operator, "<")) |
|
151 lt_op = true; |
|
152 else |
|
153 lt_op = false; |
|
154 endif |
|
155 if any (ismember (operator, ">")) |
|
156 gt_op = true; |
|
157 else |
|
158 gt_op = false; |
|
159 endif |
|
160 |
7069
|
161 ## Make sure that we don't have conflicting operators. |
6045
|
162 if (gt_op && lt_op) |
7069
|
163 error ("compare_versions: operator cannot contain both greater and less than symbols"); |
6045
|
164 elseif ((gt_op || lt_op) && not_op) |
7069
|
165 error ("compare_versions: operator cannot contain not and greater than or less than symbols"); |
6045
|
166 endif |
|
167 |
7069
|
168 ## Compare the versions (making sure that they're the same shape) |
6045
|
169 vcmp = v1n(:) - v2n(:); |
|
170 vcmp = [vcmp; (v1c - v2c)(:)]; |
|
171 if (lt_op) |
|
172 ## so that we only need to check for the output being greater than 1 |
|
173 vcmp = -vcmp; |
|
174 endif |
7069
|
175 firstdiff = find (vcmp != 0, 1); |
6045
|
176 |
7069
|
177 if (isempty (firstdiff)) |
|
178 ## They're equal. |
6045
|
179 out = equal_op; |
|
180 elseif (lt_op || gt_op) |
7069
|
181 ## They're correctly less than or greater than. |
6045
|
182 out = (vcmp(firstdiff) > 0); |
|
183 else |
7069
|
184 ## They're not correctly less than or greater than, and they're not |
|
185 ## equal. |
6045
|
186 out = false; |
|
187 endif |
|
188 |
7069
|
189 ## Reverse the output if not is given. |
6045
|
190 out = xor (not_op, out); |
7125
|
191 |
6045
|
192 endfunction |
|
193 |
|
194 ## tests |
|
195 ## test both equality symbols |
|
196 %!assert(compare_versions("1", "1", "="), true) |
|
197 ## test arbitrarily long equality |
|
198 %!assert(compare_versions("1.1.0.0.0", "1.1", "=="), true) |
|
199 %!assert(compare_versions("1", "1.1", "<"), true) |
|
200 %!assert(compare_versions("1.1", "1.1", "<="), true) |
|
201 %!assert(compare_versions("1.1", "1.1.1", "<="), true) |
|
202 %!assert(compare_versions("1.23", "1.24", "=<"), true) |
|
203 ## test different length numbers |
|
204 %!assert(compare_versions("23.2000", "23.1", ">"), true) |
|
205 %!assert(compare_versions("0.0.2", "0.0.1", ">="), true) |
|
206 %!assert(compare_versions("0.2", "0.0.100", "=>"), true) |
|
207 %!assert(compare_versions("0.1", "0.2", "!="), true) |
|
208 %!assert(compare_versions("0.1", "0.2", "~="), true) |
|
209 |
|
210 ## test alphanumeric strings |
|
211 %!assert(compare_versions("1a", "1b", "<"), true) |
|
212 %!assert(compare_versions("a", "1", "<"), true) |
|
213 %!assert(compare_versions("1a", "1b", ">"), false) |
|
214 %!assert(compare_versions("a", "1", ">"), false) |
|
215 %!assert(compare_versions("1.1.0a", "1.1.0b", "=="), false) |
|
216 %!assert(compare_versions("1.1.0a", "1.1.0b", "!="), true) |
|
217 %!assert(compare_versions("1.1.0test", "1.1.0b", "=="), false) |
|
218 %!assert(compare_versions("1.1.0test", "1.1.0test", "=="), true) |
|
219 |
|
220 ## make sure that it won't just give true output |
|
221 %!assert(compare_versions("1", "0", "="), false) |
|
222 ## test arbitrarily long equality |
|
223 %!assert(compare_versions("1.1.1.0.0", "1.1", "=="), false) |
|
224 %!assert(compare_versions("1.1", "1", "<"), false) |
|
225 %!assert(compare_versions("2", "1.1", "<="), false) |
|
226 %!assert(compare_versions("1.1.1", "1.1", "<="), false) |
|
227 %!assert(compare_versions("1.25", "1.24", "=<"), false) |
|
228 ## test different length numbers |
|
229 %!assert(compare_versions("23.2", "23.100", ">"), false) |
|
230 %!assert(compare_versions("0.0.0.2", "0.0.1", ">="), false) |
|
231 %!assert(compare_versions("0.0.20", "0.10.2", "=>"), false) |
|
232 %!assert(compare_versions("0.1", "0.1", "!="), false) |
|
233 %!assert(compare_versions("0.1", "0.1", "~="), false) |
|
234 |
|
235 ## FIXME: how do we check to make sure that it gives errors when it |
|
236 ## should |