version_recommend.py 36.4 KB
Newer Older
1 2
#!-*- coding:utf-8 -*-

S
Shinwell Hu 已提交
3 4 5
import re
import datetime
import time
L
licihua 已提交
6
from typing import List
S
Shinwell Hu 已提交
7 8 9

__ALL__ = ["VersionRecommend"]

10 11 12 13 14 15 16
"""

The base class of the version recommend, used to get the latest version and maintain version.

"""


L
licihua 已提交
17
class VersionType(object):
18
    """Base class for version recommend"""
L
licihua 已提交
19 20

    def __init__(self):
21 22 23 24 25 26 27
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
28 29
        self._version_type = None

S
Shinwell Hu 已提交
30
    def version_match(self, pkg_version):
31 32 33 34 35 36 37
        """
        Version match.

        :param pkg_version: Package version
        :returns: None
        :raises: None
        """
S
Shinwell Hu 已提交
38 39 40
        pass

    def latest_version(self, version_entry):
41 42 43 44 45 46 47
        """
        Get the latest version.

        :param version_entry: Package version list
        :returns: None
        :raises: None
        """
L
licihua 已提交
48
        version_entry.sort(reverse=True)
S
Shinwell Hu 已提交
49 50 51
        return version_entry[0]

    def maintain_version(self, version_entry, current_version, pkg_type):
52 53 54 55 56 57 58 59 60
        """
        Get the maintain version.

        :param version_entry: Package version list
        :param current_version: Current version
        :param pkg_type: Package type
        :returns : Maintain version
        :raises: None
        """
L
licihua 已提交
61
        _ = version_entry, pkg_type
62
        return current_version
S
Shinwell Hu 已提交
63

64 65 66 67 68 69 70 71 72 73 74
    def compare(self, z1, z2):
        """
        Get the max version.

        :param z1: The first version
        :param z2: The second version
        :returns 1: z1 great then z2
        :return -1: z2 great then z1
        :return 0: z1 equal then z2
        :raises: None
        """
75
        return self._compare(z1, z2)
76 77

    def _compare(self, z1, z2):
78 79 80 81 82 83 84 85 86 87
        """
        Get the max version.

        :param z1: The first version
        :param z2: The second version
        :returns 1: z1 great then z2
        :return -1: z2 great then z1
        :return 0: z1 equal then z2
        :raises: None
        """
88 89 90 91 92 93
        d1 = tuple(self._split(z1))  # 第一个参数版本号拆分,获取里面的数字/字母,得到序列
        d2 = tuple(self._split(z2))  # 第二个参数版本号拆分,获取里面的数字/字母,得到序列
        len1 = len(d1)
        len2 = len(d2)
        length = min(len1, len2)
        for index in range(length):
94
            if d1[index].isdigit() and d2[index].isdigit():
95 96 97 98
                if int(d1[index]) > int(d2[index]):
                    return 1
                elif int(d1[index]) < int(d2[index]):
                    return -1
99 100 101 102
            elif d1[index].isdigit():
                return 1
            elif d2[index].isdigit():
                return -1
103 104 105 106 107 108 109
            else:
                if d1[index] > d2[index]:
                    return 1
                elif d1[index] < d2[index]:
                    return -1
        if len1 > len2:
            return 1
L
LeoFang 已提交
110
        elif len1 < len2:
111
            return -1
L
LeoFang 已提交
112 113
        else:
            return 0
114

S
Shinwell Hu 已提交
115
    def get_version_mode(self):
116 117 118 119 120 121 122
        """
        Get the version mode.

        :param: None
        :returns: Version type
        :raises: None
        """
S
Shinwell Hu 已提交
123 124
        return self._version_type

125
    def _split(self, x):
126 127 128 129 130 131 132
        """
        Split the input args.

        :param x: Input args
        :returns: The split result
        :raises: None
        """
133
        for f, s in re.findall(r'([\d]+)|([^\d.-]+)', x):
134 135 136 137 138 139 140
            if f:
                float(f)
                yield f
            else:
                yield s


L
licihua 已提交
141
class VersionTypeXYZW(VersionType):
142
    """Version type Class for x.y.z.w"""
143

S
Shinwell Hu 已提交
144
    def version_match(self, pkg_version):
145 146 147 148 149 150 151 152
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
153
        version = pkg_version.strip()
L
licihua 已提交
154
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
155 156 157 158 159 160 161 162 163 164 165 166 167
        if len(digital_list) != 4:  # 通过 '.'分割后,应该剩下4位
            return False
        if len(digital_list[0]) > 2:  # 第一位版本号不应该大于2位
            return False
        if len(digital_list[1]) > 2:  # 第二位版本号不应该大于2位
            return False
        if len(digital_list[2]) > 3:  # 第三位版本号不应该大于3位
            return False
        if len(digital_list[3]) > 1:  # 第四位版本号不应该大于1位
            return False
        return True

    def maintain_version(self, version_entry, current_version, pkg_type):
168 169 170 171 172 173 174 175 176
        """
        Get the maintain version.

        :param version_entry: Package version list
        :param current_version: Current version
        :param pkg_type: Package type
        :returns : Maintain version
        :raises: None
        """
L
licihua 已提交
177
        if len(version_entry) == 1:  # 仅一个版本,当前即为最新版本
L
licihua 已提交
178 179 180 181 182 183 184 185
            return version_entry[0]

        version_candidate = []
        version_digital = re.split(r'[._-]', current_version)  # 将版本号做拆分
        xyz = version_digital[0:3]
        for version in version_entry:
            version_temp = re.split(r'[._-]', version)
            if version_digital[0:3] == version_temp[0:3]:  # 如果版本号与当前版本前三位一致,说明是维护分支版本
L
licihua 已提交
186
                version_candidate.append(version_temp)  # 将同特性版本的子版本挑选出来
L
licihua 已提交
187 188 189 190 191 192 193 194 195

        if len(version_candidate) == 1:
            return '.'.join(version_candidate[0])

        w = '0'
        for version in version_candidate:
            if len(version) <= 3:
                continue

196
            if self._compare(version[3], w) > 0:
L
licihua 已提交
197
                w = version[3]
198

L
licihua 已提交
199 200 201 202
        xyz.append(w)
        return '.'.join(xyz)

    def latest_version(self, version_entry):
203 204 205 206 207 208 209
        """
        Get latest version.

        :param version_entry: Package version list
        :returns: latest version
        :raises: None
        """
L
licihua 已提交
210
        if len(version_entry) == 1:  # 仅一个版本,当前即为最新版本
L
licihua 已提交
211 212
            return version_entry[0]
        version_list = []
S
Shinwell Hu 已提交
213
        for version in version_entry:
L
licihua 已提交
214
            version_list.append(re.split(r'[._-]', version))  # 将 version 拆分为列表,方便后续比较
L
licihua 已提交
215
        x = '0'
L
licihua 已提交
216
        for version in version_list:  # 第一轮比较取出最大的第一位
S
Shinwell Hu 已提交
217 218 219
            if int(version[0]) > 1000: # consider it an useless exception
                continue

220
            if self._compare(x, version[0]) < 0:
L
licihua 已提交
221 222 223
                x = version[0]

        version_candidate = []
L
licihua 已提交
224
        for version in version_list:  # 将第一位最大的列入候选列表,准备第二位比较
L
licihua 已提交
225 226 227 228 229 230 231 232
            if x == version[0]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        version_list = version_candidate[:]
        y = '0'
L
licihua 已提交
233 234
        for version in version_list:  # 第二轮比较取出最大的第二位
            if len(version) <= 1:  # 过滤仅一位的版本号
L
licihua 已提交
235
                continue
236
            if self._compare(y, version[1]) < 0:
L
licihua 已提交
237 238 239
                y = version[1]

        version_candidate.clear()
L
licihua 已提交
240
        for version in version_list:  # 将第二位最大的列入候选列表,准备第三位比较
L
licihua 已提交
241 242 243 244 245 246 247 248
            if y == version[1]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        z = '0'
        version_list = version_candidate[:]
L
licihua 已提交
249 250
        for version in version_list:  # 第三轮比较取出最大的第三位
            if len(version) <= 2:  # 过滤仅二位的版本号
L
licihua 已提交
251
                continue
252
            if self._compare(z, version[2]) < 0:
L
licihua 已提交
253 254 255
                z = version[2]

        version_candidate.clear()
L
licihua 已提交
256 257
        for version in version_list:  # 最后一位最大版本必须惟一,直接返回结果
            if len(version) <= 2:  # 过滤仅二位的版本号
L
licihua 已提交
258 259 260 261 262 263 264 265 266 267 268 269
                continue
            if z == version[2]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        w = '0'
        version_list = version_candidate[:]
        for version in version_list:  # 最后一位最大版本必须惟一,直接返回结果
            if len(version) <= 3:  # 过滤仅三位的版本号
                continue
270
            if self._compare(w, version[3]) < 0:
L
licihua 已提交
271 272
                w = version[3]

L
licihua 已提交
273
        for version in version_list:  # 最后一位最大版本必须惟一,直接返回结果
L
licihua 已提交
274 275 276 277 278 279
            if len(version) <= 3:  # 过滤仅三位的版本号
                continue
            if w == version[3]:
                return '.'.join(version)

        return ''
S
Shinwell Hu 已提交
280 281

    def __init__(self):
282 283 284 285 286 287 288
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
289
        super().__init__()
S
Shinwell Hu 已提交
290 291 292
        self._version_type = 'x.y.z.w'


L
licihua 已提交
293
class VersionTypeXYZ(VersionType):
294
    """Version type Class for x.y.z"""
L
licihua 已提交
295

S
Shinwell Hu 已提交
296
    def version_match(self, pkg_version):
297 298 299 300 301 302 303 304
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
305
        version = pkg_version.strip()
L
licihua 已提交
306
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
307 308 309 310
        if len(digital_list) != 3:  # 通过 '.'分割后,应该剩下3位
            return False
        if len(digital_list[0]) > 2:  # 第一位版本号不应该大于2位
            return False
L
licihua 已提交
311
        if len(digital_list[1]) > 3:  # 第二位版本号不应该大于3位
S
Shinwell Hu 已提交
312 313 314 315 316 317
            return False
        if len(digital_list[2]) > 3:  # 第三位版本号不应该大于3位
            return False
        return True

    def maintain_version(self, version_entry, current_version, pkg_type):
318 319 320 321 322 323 324 325 326
        """
        Get the maintain version.

        :param version_entry: Package version list
        :param current_version: Current version
        :param pkg_type: Package type
        :returns : Maintain version
        :raises: None
        """
L
licihua 已提交
327
        if len(version_entry) == 1:  # 仅一个版本,当前即为最新版本
328 329 330
            return version_entry[0]

        version_candidate = []
L
licihua 已提交
331
        version_digital = re.split(r'[._-]', current_version)  # 将版本号做拆分
332
        xy = version_digital[0:2]
S
Shinwell Hu 已提交
333
        for version in version_entry:
L
licihua 已提交
334
            version_temp = re.split(r'[._-]', version)
335
            if version_digital[0:2] == version_temp[0:2]:  # 如果版本号与当前版本前两位一致,说明是维护分支版本
L
licihua 已提交
336
                version_candidate.append(version_temp)  # 将同特性版本的子版本挑选出来
337 338 339 340 341 342 343 344 345

        if len(version_candidate) == 1:
            return '.'.join(version_candidate[0])

        z = '0'
        for version in version_candidate:
            if len(version) <= 2:
                continue

346
            if self._compare(version[2], z) > 0:
347 348 349 350 351 352
                z = version[2]

        xy.append(z)
        return '.'.join(xy)

    def latest_version(self, version_entry):
353 354 355 356 357 358 359
        """
        Get latest version.

        :param version_entry: Package version list
        :returns: latest version
        :raises: None
        """
L
licihua 已提交
360
        if len(version_entry) == 1:  # 仅一个版本,当前即为最新版本
361 362 363
            return version_entry[0]
        version_list = []
        for version in version_entry:
L
licihua 已提交
364
            version_list.append(re.split(r'[._-]', version))  # 将 version 拆分为列表,方便后续比较
365
        x = '0'
L
licihua 已提交
366
        for version in version_list:  # 第一轮比较取出最大的第一位
S
Shinwell Hu 已提交
367 368
            if int(version[0]) > 1000: # consider it an useless exception
                continue
369
            if self._compare(x, version[0]) < 0:
370 371 372
                x = version[0]

        version_candidate = []
L
licihua 已提交
373
        for version in version_list:  # 将第一位最大的列入候选列表,准备第二位比较
374 375 376 377 378 379 380 381
            if x == version[0]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        version_list = version_candidate[:]
        y = '0'
L
licihua 已提交
382 383
        for version in version_list:  # 第二轮比较取出最大的第二位
            if len(version) <= 1:  # 过滤仅一位的版本号
384
                continue
385
            if self._compare(y, version[1]) < 0:
386
                y = version[1]
387

388
        version_candidate.clear()
L
licihua 已提交
389
        for version in version_list:  # 将第二位最大的列入候选列表,准备第三位比较
390 391 392 393 394 395 396 397
            if y == version[1]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        z = '0'
        version_list = version_candidate[:]
L
licihua 已提交
398 399
        for version in version_list:  # 第三轮比较取出最大的第三位
            if len(version) <= 2:  # 过滤仅二位的版本号
400
                continue
401
            if self._compare(z, version[2]) < 0:
402
                z = version[2]
403

L
licihua 已提交
404
        for version in version_list:  # 最后一位最大版本必须惟一,直接返回结果
405 406 407 408 409 410
            if len(version) <= 2:  # 过滤仅二位的版本号
                continue
            if z == version[2]:
                return '.'.join(version)

        return ''
S
Shinwell Hu 已提交
411 412

    def __init__(self):
413 414 415 416 417 418 419
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
420
        super().__init__()
S
Shinwell Hu 已提交
421 422 423
        self._version_type = 'x.y.z'


L
licihua 已提交
424
class VersionTypeXY(VersionType):
425
    """Version type Class for x.y"""
L
licihua 已提交
426

S
Shinwell Hu 已提交
427
    def version_match(self, pkg_version):
428 429 430 431 432 433 434 435
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
436
        version = pkg_version.strip()
L
licihua 已提交
437
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
438 439 440 441 442 443 444 445 446
        if len(digital_list) != 2:  # 通过 '.'分割后,应该剩下2位
            return False
        if len(digital_list[0]) > 2:  # 第一位版本号不应该大于2位
            return False
        if len(digital_list[1]) > 3:  # 第二位版本号不应该大于2位
            return False
        return True

    def __init__(self):
447 448 449 450 451 452 453
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
454
        super().__init__()
S
Shinwell Hu 已提交
455 456
        self._version_type = 'x.y'

457
    def latest_version(self, version_entry):
458 459 460 461 462 463 464
        """
        Get latest version.

        :param version_entry: Package version list
        :returns: latest version
        :raises: None
        """
L
licihua 已提交
465
        if len(version_entry) == 1:  # 仅一个版本,当前即为最新版本
466 467 468
            return version_entry[0]
        version_list = []
        for version in version_entry:
L
licihua 已提交
469
            version_list.append(re.split(r'[._-]', version))  # 将 version 拆分为列表,方便后续比较
470
        x = '0'
L
licihua 已提交
471
        for version in version_list:  # 第一轮比较取出最大的第一位
S
Shinwell Hu 已提交
472 473 474
            if int(version[0]) > 1000: # consider it an useless exception
                continue

475
            if self._compare(x, version[0]) < 0:
476
                x = version[0]
477

478
        version_candidate = []
L
licihua 已提交
479
        for version in version_list:  # 将第一位最大的列入候选列表,准备第二位比较
480 481 482 483 484 485 486 487
            if x == version[0]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        version_list = version_candidate[:]
        y = '0'
L
licihua 已提交
488 489
        for version in version_list:  # 第二轮比较取出最大的第二位
            if len(version) <= 1:  # 过滤仅一位的版本号
490
                continue
491
            if self._compare(y, version[1]) < 0:
492
                y = version[1]
493

494
        version_candidate.clear()
L
licihua 已提交
495 496
        for version in version_list:  # x.y 版本类型中会小概率出现三位版本号,需要将第二位最大的列入候选列表,准备第三位比较
            if len(version) <= 1:  # 过滤仅一位的版本号
L
licihua 已提交
497
                continue
498 499 500 501 502 503 504 505
            if y == version[1]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        z = '0'
        version_list = version_candidate[:]
L
licihua 已提交
506 507
        for version in version_list:  # 第三轮比较取出最大的第三位
            if len(version) <= 2:  # 过滤仅二位的版本号
508
                continue
509
            if self._compare(z, version[2]) < 0:
510 511
                z = version[2]

L
licihua 已提交
512
        for version in version_list:  # 最后一位最大版本必须惟一,直接返回结果
513 514 515 516 517 518 519 520
            if len(version) <= 2:  # 过滤仅二位的版本号
                continue
            if z == version[2]:
                return '.'.join(version)

        return ''

    def maintain_version(self, version_entry, current_version, pkg_type):
521 522 523 524 525 526 527 528 529
        """
        Get the maintain version.

        :param version_entry: Package version list
        :param current_version: Current version
        :param pkg_type: Package type
        :returns : Maintain version
        :raises: None
        """
530
        version_candidate = []
L
licihua 已提交
531
        version_digital = re.split(r'[._-]', current_version)  # 将版本号做拆分
L
licihua 已提交
532
        x = [version_digital[0]]
533
        for version in version_entry:
L
licihua 已提交
534
            version_temp = re.split(r'[._-]', version)
535
            if version_digital[0] == version_temp[0]:  # 如果版本号与当前版本前两位一致,说明是维护分支版本
L
licihua 已提交
536
                version_candidate.append(version_temp)  # 将同特性版本的子版本挑选出来
537 538 539 540 541 542 543 544 545

        if len(version_candidate) == 1:
            return '.'.join(version_candidate[0])

        y = '0'
        for version in version_candidate[0:]:
            if len(version) <= 1:
                continue

546
            if self._compare(version[1], y) > 0:
547 548 549
                y = version[1]
        x.append(y)
        return '.'.join(x)
S
Shinwell Hu 已提交
550

L
licihua 已提交
551 552

class VersionTypeX(VersionType):
553
    """Version type Class for x"""
L
licihua 已提交
554

S
Shinwell Hu 已提交
555
    def version_match(self, pkg_version):
556 557 558 559 560 561 562 563
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
564
        version = pkg_version.strip()
L
licihua 已提交
565
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
566 567 568 569 570 571
        if len(digital_list) != 1:  # 通过 '.'分割后,应该剩下1位
            return False
        if len(digital_list[0]) > 3:  # 第一位版本号不应该大于3位
            return False
        return True

572
    def latest_version(self, version_entry):
573 574 575 576 577 578 579
        """
        Get latest version.

        :param version_entry: Package version list
        :returns: latest version
        :raises: None
        """
L
licihua 已提交
580
        if 1 == len(version_entry):  # 仅一个版本,当前即为最新版本
581
            return version_entry[0]
L
licihua 已提交
582
        version_list: List[List[str]] = []
583
        for version in version_entry:
L
licihua 已提交
584
            version_list.append(re.split(r'[._-]', version))  # 将 version 拆分为列表,方便后续比较
585 586
        x = '0'
        for version in version_list:  # 第一轮比较取出最大的第一位
S
Shinwell Hu 已提交
587 588 589
            if int(version[0]) > 1000: # consider it an useless exception
                continue

590
            if self._compare(x, version[0]) < 0:
591 592 593 594 595 596 597 598 599 600 601 602 603 604 605
                x = version[0]

        version_candidate = []
        for version in version_list:  # 将第一位最大的列入候选列表,准备第二位比较
            if x == version[0]:
                version_candidate.append(version)

        if len(version_candidate) == 1:  # 仅一个版本,候选即为最新版本
            return '.'.join(version_candidate[0])

        version_list = version_candidate[:]
        y = '0'
        for version in version_list:  # 第二轮比较取出最大的第二位
            if len(version) <= 1:  # 过滤仅一位的版本号
                continue
606
            if self._compare(y, version[1]) < 0:
607 608 609 610 611 612
                y = version[1]

        x.append(y)
        return '.'.join(x)

    def maintain_version(self, version_entry, current_version, pkg_type):
613 614 615 616 617 618 619 620 621
        """
        Get the maintain version.

        :param version_entry: Package version list
        :param current_version: Current version
        :param pkg_type: Package type
        :returns : Maintain version
        :raises: None
        """
622
        version_candidate = []
L
licihua 已提交
623
        version_digital = re.split(r'[._-]', current_version)  # 将版本号做拆分
L
licihua 已提交
624
        x = [version_digital[0]]
625
        for version in version_entry:
L
licihua 已提交
626
            version_temp = re.split(r'[._-]', version)
627
            if version_digital[0] == version_temp[0]:  # 如果版本号与当前版本前两位一致,说明是维护分支版本
L
licihua 已提交
628
                version_candidate.append(version_temp)  # 将同特性版本的子版本挑选出来
629 630 631 632 633 634 635 636 637

        if len(version_candidate) == 1:
            return '.'.join(version_candidate[0])

        y = '0'
        for version in version_candidate[0:]:
            if len(version) <= 1:
                continue

638
            if self._compare(version[1], y) > 0:
639 640 641
                y = version[1]
        x.append(y)
        return '.'.join(x)
L
licihua 已提交
642

S
Shinwell Hu 已提交
643
    def __init__(self):
644 645 646 647 648 649 650
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
651
        super().__init__()
S
Shinwell Hu 已提交
652 653 654
        self._version_type = 'x'


L
licihua 已提交
655
class VersionTypeYyyyXY(VersionType):
656
    """Version type Class for yyyy.x.y"""
L
licihua 已提交
657

S
Shinwell Hu 已提交
658
    def version_match(self, pkg_version):
659 660 661 662 663 664 665 666
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
667
        version = pkg_version.strip()
L
licihua 已提交
668
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
669 670 671 672 673
        if len(digital_list) != 3:  # 通过 '.'分割后,应该剩下3位
            return False

        if len(digital_list[0]) != 4:  # 第一位为发布年份,位数为4位
            return False
L
licihua 已提交
674

S
Shinwell Hu 已提交
675 676 677 678 679 680 681
        year = int(digital_list[0])
        if year < 2000 or year > datetime.datetime.now().year:  # 软件发布时间应该大于2000 年,小于当前时间
            return False

        if len(digital_list[1]) > 2:  # 第二位版本号不应该大于2位
            return False

L
licihua 已提交
682
        if str(int(digital_list[1])) != digital_list[1]:  # 版本类型为数字,且非0 开头
L
licihua 已提交
683 684
            return False

S
Shinwell Hu 已提交
685 686
        if len(digital_list[2]) > 2:  # 第三位版本号不应该大于2位
            return False
L
licihua 已提交
687

L
licihua 已提交
688
        if str(int(digital_list[2])) != digital_list[2]:  # 版本类型为数字,且非0 开头
L
licihua 已提交
689 690
            return False

S
Shinwell Hu 已提交
691 692 693
        return True

    def __init__(self):
694 695 696 697 698 699 700
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
701
        super().__init__()
S
Shinwell Hu 已提交
702 703 704
        self._version_type = 'yyyy.x.y'


L
licihua 已提交
705
class VersionTypeYyyyX(VersionType):
706
    """Version type Class for yyyy.x"""
L
licihua 已提交
707

S
Shinwell Hu 已提交
708
    def version_match(self, pkg_version):
709 710 711 712 713 714 715 716
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
717
        version = pkg_version.strip()
L
licihua 已提交
718
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
719 720 721 722 723 724 725 726 727 728 729 730 731 732
        if len(digital_list) != 2:  # 通过 '.'分割后,应该剩下2位
            return False

        if len(digital_list[0]) != 4:  # 第一位为发布年份,位数为4位
            return False
        year = int(digital_list[0])
        if year < 2000 or year > datetime.datetime.now().year:  # 软件发布时间应该大于2000 年,小于当前时间
            return False

        if len(digital_list[1]) > 2:  # 第二位版本号不应该大于2位
            return False
        return True

    def __init__(self):
733 734 735 736 737 738 739
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
740
        super().__init__()
S
Shinwell Hu 已提交
741 742 743
        self._version_type = 'yyyy.x'


L
licihua 已提交
744
class VersionTypeYyyyW(VersionType):
745
    """Version type Class for yyyyw"""
L
licihua 已提交
746

L
licihua 已提交
747
    def version_match(self, pkg_version):
748 749 750 751 752 753 754 755
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
L
licihua 已提交
756 757 758
        version = pkg_version.strip()
        if len(version) != 5:  # 共5 位
            return False
L
licihua 已提交
759
        if not str(version[0:4]).isdigit():  # 前四位为年份数字
L
licihua 已提交
760 761 762 763 764 765 766 767 768
            return False

        year = int(version[0:4])
        if year < 2000 or year > datetime.datetime.now().year:  # 软件发布时间应该大于2000 年,小于当前时间
            return False

        return True

    def __init__(self):
769 770 771 772 773 774 775
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
776
        super().__init__()
L
licihua 已提交
777 778
        self._version_type = 'yyyyw'

L
licihua 已提交
779 780

class VersionTypeYyyyMmDd(VersionType):
781
    """Version type Class for yyyy.mm.dd"""
L
licihua 已提交
782

S
Shinwell Hu 已提交
783
    def version_match(self, pkg_version):
784 785 786 787 788 789 790 791
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
792
        version = pkg_version.strip()
L
licihua 已提交
793
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809
        if len(digital_list) != 3:  # 通过 '.'分割后,应该剩下3位
            return False

        if len(digital_list[0]) != 4:  # 第一位为发布年份,位数为4位
            return False

        if int(digital_list[1]) > 12 or int(digital_list[1]) == 0:  # 第二位为发布月份,小于12
            return False

        if int(digital_list[2]) > 31 or int(digital_list[2]) == 0:  # 第三位为发布日期,小于31
            return False

        # 判断日期是否为合法日期
        try:
            if '_' in version:
                d_time = time.mktime(time.strptime(version, "%Y_%m_%d"))
L
licihua 已提交
810
            elif '-' in version:
L
licihua 已提交
811 812
                d_time = time.mktime(time.strptime(version, "%Y-%m-%d"))
            elif '.' in version:
S
Shinwell Hu 已提交
813
                d_time = time.mktime(time.strptime(version, "%Y.%m.%d"))
L
licihua 已提交
814 815
            else:
                return False
S
Shinwell Hu 已提交
816 817 818 819 820 821 822

            now_str = datetime.datetime.now().strftime('%Y-%m-%d')
            end_time = time.mktime(time.strptime(now_str, '%Y-%m-%d'))
            if d_time > end_time:  # 判断日期是否大于当前日期
                return False
            else:
                return True
L
licihua 已提交
823
        except ValueError as e:  # 时间格式非法
824
            _ = e
S
Shinwell Hu 已提交
825 826 827 828
            print('Time foramt failed %s.', version)
            return False

    def __init__(self):
829 830 831 832 833 834 835
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
836
        super().__init__()
S
Shinwell Hu 已提交
837 838 839
        self._version_type = 'yyyy.mm.dd'


L
licihua 已提交
840
class VersionTypeYyyyMm(VersionType):
841
    """Version type Class for yyyy.mm"""
L
licihua 已提交
842

S
Shinwell Hu 已提交
843
    def version_match(self, pkg_version):
844 845 846 847 848 849 850 851
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
852
        version = pkg_version.strip()
L
licihua 已提交
853
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872
        if len(digital_list) != 2:  # 通过 '.'分割后,应该剩下2位
            return False

        if len(digital_list[0]) != 4:  # 第一位为发布年份,位数为4位
            return False
        year = int(digital_list[0])
        if year < 2000 or year > datetime.datetime.now().year:  # 软件发布时间应该大于2000 年,小于当前时间
            return False

        if int(digital_list[1]) > 12 or int(digital_list[1]) == 0:  # 第二位为发布月份,小于12
            return False

        if year == datetime.datetime.now().year and \
                int(digital_list[1]) > datetime.datetime.now().month:
            return False

        return True

    def __init__(self):
873 874 875 876 877 878 879
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
880
        super().__init__()
S
Shinwell Hu 已提交
881 882
        self._version_type = 'yyyy.mm'

L
licihua 已提交
883

884 885
class VersionTypeYyyymm(VersionType):
    """Version type Class for yyyy.mm"""
L
licihua 已提交
886

887
    def version_match(self, pkg_version):
888 889 890 891 892 893 894 895
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
896
        version = pkg_version.strip()
L
licihua 已提交
897
        if len(version) != 6:  # 长度为6
898 899
            return False

L
licihua 已提交
900
        if not version.isdigit():  # 时间格式为数字
901 902
            return False

L
licihua 已提交
903
        digital_list = [version[0:4], version[4:]]
904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920

        if len(digital_list[0]) != 4:  # 第一位为发布年份,位数为4位
            return False
        year = int(digital_list[0])
        if year < 2000 or year > datetime.datetime.now().year:  # 软件发布时间应该大于2000 年,小于当前时间
            return False

        if int(digital_list[1]) > 12 or int(digital_list[1]) == 0:  # 第二位为发布月份,小于12
            return False

        if year == datetime.datetime.now().year and \
                int(digital_list[1]) > datetime.datetime.now().month:
            return False

        return True

    def __init__(self):
921 922 923 924 925 926 927
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
928
        super().__init__()
929
        self._version_type = 'yyyymm'
S
Shinwell Hu 已提交
930

L
licihua 已提交
931 932

class VersionTypeXYymmZ(VersionType):
933
    """Version type Class for x.yymm.z"""
S
Shinwell Hu 已提交
934
    def version_match(self, pkg_version):
935 936 937 938 939 940 941 942
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
943
        version = pkg_version.strip()
L
licihua 已提交
944
        digital_list = re.split(r'[._-]', version)
S
Shinwell Hu 已提交
945 946 947 948 949 950
        if len(digital_list) != 3:  # 通过 '.'分割后,应该剩下3位
            return False

        if len(digital_list[0]) > 2:  # 第一位为主版本号,小于2位
            return False
        # 将年月拆分后分别判断
L
licihua 已提交
951
        if len(digital_list[1]) != 4:  # 年月等于4位
952
            return False
S
Shinwell Hu 已提交
953 954 955 956
        year = str(digital_list[1][:2])
        month = str(digital_list[1][-2:])
        if year > datetime.datetime.now().year[-2:]:  # 年份不能大于当前年份,不用考虑20000 年前的情况
            return False
L
licihua 已提交
957
        if month > '12' or month == '0':
S
Shinwell Hu 已提交
958 959 960 961 962 963
            return False
        if len(digital_list[2]) > 2:  # 迭代号不大于2位
            return False
        return True

    def __init__(self):
964 965 966 967 968 969 970
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
971
        super().__init__()
S
Shinwell Hu 已提交
972 973 974
        self._version_type = 'x.yymm.z'


L
licihua 已提交
975
class VersionTypeYyyymmdd(VersionType):
976
    """Version type Class for yyyymmdd"""
L
licihua 已提交
977

S
Shinwell Hu 已提交
978
    def version_match(self, pkg_version):
979 980 981 982 983 984 985 986
        """
        Version match.

        :param pkg_version: Package version
        :returns True:  Version match success
        :returns False: version match fail
        :raises: None
        """
S
Shinwell Hu 已提交
987 988 989 990 991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008
        version = pkg_version.strip()
        if len(version) != 8:  # 日期长度满足 8 位要求
            return False
        if not version.isdigit():  # 连续数字,没有其他分别符号
            return False
        digital_list = [version[:4], version[4:6], version[6:]]

        if int(digital_list[1]) > 12 or int(digital_list[1]) == 0:  # 第二位为发布月份,小于12
            return False

        if int(digital_list[2]) > 31 or int(digital_list[2]) == 0:  # 第三位为发布日期,小于31
            return False

        # 判断日期是否为合法日期
        try:
            d_time = time.mktime(time.strptime(version, "%Y%m%d"))
            now_str = datetime.datetime.now().strftime('%Y-%m-%d')
            end_time = time.mktime(time.strptime(now_str, '%Y-%m-%d'))
            if d_time > end_time:  # 判断日期是否大于当前日期
                return False
            else:
                return True
L
licihua 已提交
1009
        except ValueError as e:  # 时间格式非法
1010
            _ = e
S
Shinwell Hu 已提交
1011 1012 1013 1014
            print('Time format failed %s,', version)
            return False

    def __init__(self):
1015 1016 1017 1018 1019 1020 1021
        """
        Initialize.

        :param None: No parameter
        :returns: None
        :raises: None
        """
L
licihua 已提交
1022
        super().__init__()
S
Shinwell Hu 已提交
1023 1024 1025
        self._version_type = 'yyyymmdd'


L
licihua 已提交
1026
class VersionRecommend(object):
1027
    """Version recommend Class for open source"""
S
Shinwell Hu 已提交
1028
    def __init__(self, version_entry, current_version, pkg_type):
1029 1030 1031 1032 1033 1034 1035 1036
        """
        Initialize.

        :param version_entry: The version list of open source
        :param current_version: The current version  of open source
        :returns: None
        :raises: None
        """
L
licihua 已提交
1037
        self.latest_version = current_version  # 提供初值,避免 current_version 为空导致后面出现异常
L
licihua 已提交
1038
        self.maintain_version = current_version
1039 1040 1041 1042 1043 1044
        self.version_type = self._version_match(version_entry)
        if self.version_type is None:
            print('version type is None:', current_version)
            return

        print('version type = ', self.version_type.get_version_mode())
S
Shinwell Hu 已提交
1045 1046 1047
        self.latest_version = self._get_latest_version(version_entry)
        self.maintain_version = self._get_maintain_version(version_entry, current_version, pkg_type)

1048
    def _version_match(self, version_entry):
1049 1050
        """
        Version match function.
1051

1052 1053 1054 1055
        :param version_entry: The version list of open source
        :returns: The method input version list
        :raises: None
        """
L
licihua 已提交
1056 1057 1058 1059 1060 1061 1062 1063
        version_method = {VersionTypeXYZW(): 0,
                          VersionTypeXYZ(): 0,
                          VersionTypeXY(): 0,
                          VersionTypeX(): 0,
                          VersionTypeYyyyXY(): 0,
                          VersionTypeYyyyX(): 0,
                          VersionTypeYyyyW(): 0,
                          VersionTypeYyyyMmDd(): 0,
1064
                          VersionTypeYyyymm(): 0,
L
licihua 已提交
1065 1066
                          VersionTypeXYymmZ(): 0,
                          VersionTypeYyyymmdd(): 0}
1067 1068 1069 1070
        if not version_entry:
            return None
        for version in version_entry[:]:
            if not self._version_valid(version):
L
licihua 已提交
1071
                version_entry.remove(version)  # 删除非法版本号
1072 1073 1074 1075 1076 1077 1078
                continue
            for method, count in version_method.items():
                if method.version_match(version):
                    version_method[method] = count + 1

        # 解决多版本类型问题,选取类型最多的作为匹配,这个处理不是最优方案,需要改进
        method = max(version_method, key=lambda x: version_method[x])
L
licihua 已提交
1079

1080 1081 1082 1083 1084
        if version_method[method] == 0:
            return None
        else:
            return method

1085 1086 1087 1088 1089 1090 1091 1092 1093 1094
    @staticmethod
    def _version_valid(version):
        """
        Version valid check.

        :param version: The version of open source
        :returns True: valid version
        :returns False: invalid version
        :raises: None
        """
L
licihua 已提交
1095 1096
        m = re.match("^[0-9a-zA-Z._-]*$", version)
        if m is None:  # 版本号应该是 数字/小写字母/下划线/. 组成
1097 1098 1099
            return False

        m = re.match('^[0-9].*', version)
L
licihua 已提交
1100
        if m is None:  # 版本号应该是数字开头
1101 1102
            return False

L
Leo Fang 已提交
1103
        m = re.search(r'[ab]\d', version)
1104 1105 1106
        if not m is None:
            return False

L
licihua 已提交
1107 1108 1109 1110 1111 1112 1113 1114 1115 1116
        if 'rc' in version \
                or 'RC' in version \
                or 'dev' in version \
                or 'beta' in version \
                or 'Beta' in version \
                or 'BETA' in version \
                or 'alpha' in version \
                or 'pl' in version \
                or 'pre' in version \
                or 'PRE' in version \
L
licihua 已提交
1117
                or 'bp' in version:  # 仅获取正式版本
L
licihua 已提交
1118 1119 1120
            return False

        if 'ubuntu' in version or 'fedora' in version:  # 去掉厂家专用版本号
1121 1122 1123
            return False

        return True
S
Shinwell Hu 已提交
1124 1125

    def _get_latest_version(self, version_entry):
1126 1127 1128 1129
        if self.version_type is None:
            return ''
        else:
            return self.version_type.latest_version(version_entry)
S
Shinwell Hu 已提交
1130 1131

    def _get_maintain_version(self, version_entry, current_version, pkg_type):
1132 1133 1134 1135
        if self.version_type is None:
            return ''
        else:
            return self.version_type.maintain_version(version_entry, current_version, pkg_type)