__init__.py 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. # ============================================================================
  2. # This file is part of Pwman3.
  3. #
  4. # Pwman3 is free software; you can redistribute it and/or modify
  5. # it under the terms of the GNU General Public License, version 2
  6. # as published by the Free Software Foundation;
  7. #
  8. # Pwman3 is distributed in the hope that it will be useful,
  9. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. # GNU General Public License for more details.
  12. #
  13. # You should have received a copy of the GNU General Public License
  14. # along with Pwman3; if not, write to the Free Software
  15. # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  16. # ============================================================================
  17. # Copyright (C) 2012 Oz Nahum Tiram <nahumoz@gmail.com>
  18. # ============================================================================
  19. # Copyright (C) 2006 Ivan Kelly <ivan@ivankelly.net>
  20. # ============================================================================
  21. import argparse
  22. import urllib.request
  23. import os
  24. import pkg_resources
  25. import re
  26. import string
  27. import shutil
  28. import sys
  29. from pwman.util import config
  30. from pwman.data.factory import check_db_version
  31. try:
  32. import cryptography # noqa
  33. has_cryptography = True
  34. except ImportError:
  35. has_cryptography = False
  36. appname = "pwman3"
  37. try:
  38. version = pkg_resources.get_distribution('pwman3').version
  39. except pkg_resources.DistributionNotFound: # pragma: no cover
  40. version = "0.9.11"
  41. class PkgMetadata(object):
  42. def __init__(self):
  43. p = pkg_resources.get_distribution('pwman3')
  44. f = open(os.path.join(p.location+'-info', 'PKG-INFO'))
  45. lines = f.readlines()
  46. self.summary = lines[3].split(':')[-1].strip()
  47. self.description = ''.join(map(string.strip, lines[9:14]))
  48. self.author_email = lines[6].split(':')[-1].strip()
  49. self.author = lines[5].split(':')[-1].strip()
  50. self.home_page = lines[4].split(':')[-1].strip()
  51. try:
  52. pkg_meta = PkgMetadata()
  53. website = pkg_meta.home_page
  54. author = pkg_meta.author
  55. authoremail = pkg_meta.author_email
  56. description = pkg_meta.summary
  57. long_description = pkg_meta.description
  58. except IOError as E:
  59. # this should only happen once when installing the package
  60. description = "a command line password manager with support for multiple databases." # noqa
  61. website = 'http://pwman3.github.io/pwman3/'
  62. def parser_options(formatter_class=argparse.HelpFormatter): # pragma: no cover
  63. parser = argparse.ArgumentParser(prog='pwman3',
  64. description=description,
  65. formatter_class=formatter_class)
  66. parser.add_argument('-c', '--config', dest='cfile',
  67. default=os.path.join(
  68. config.find_config_dir(
  69. 'pwman')[0], 'config'),
  70. help='cofiguration file to read')
  71. parser.add_argument('-d', '--database', dest='dbase')
  72. parser.add_argument('-i', '--import', nargs=2, dest='file_delim',
  73. help="Specify the file name and the delimeter type")
  74. subparsers = parser.add_subparsers(help='commands', dest="cmd")
  75. printer = subparsers.add_parser('p', help='print password entry')
  76. printer.add_argument("node", type=int)
  77. copy = subparsers.add_parser('cp', help='copy password entry to clipboard')
  78. copy.add_argument("node", type=int)
  79. version = subparsers.add_parser('version', help='version')
  80. version.add_argument("--latest", action='store_true')
  81. return parser
  82. def get_conf(args):
  83. for dir in config.find_config_dir('pwman'):
  84. if not os.path.isdir(dir): # pragma: no cover
  85. os.makedirs(dir, exist_ok=True)
  86. configp = config.Config(args.cfile, config.default_config)
  87. return configp
  88. def set_xsel(configp, OSX):
  89. if not OSX:
  90. xselpath = shutil.which("xsel") or ""
  91. configp.set_value("Global", "xsel", xselpath)
  92. elif OSX:
  93. pbcopypath = shutil.which("pbcopy") or ""
  94. configp.set_value("Global", "xsel", pbcopypath)
  95. def set_umask(configp):
  96. umask = configp.get_value("Global", "umask")
  97. if re.search(r'^\d{4}$', umask):
  98. os.umask(int(umask))
  99. def set_db(args, configp):
  100. if args.dbase:
  101. configp.set_value("Database", "dburi", args.dbase)
  102. configp.set_value("Global", "save", "False")
  103. def get_conf_options(args, OSX):
  104. configp = get_conf(args)
  105. xselpath = configp.get_value("Global", "xsel")
  106. if not xselpath: # pragma: no cover
  107. set_xsel(configp, OSX)
  108. set_db(args, configp)
  109. set_umask(configp)
  110. dburi = configp.get_value("Database", "dburi")
  111. return xselpath, dburi, configp
  112. def get_db_version(config, args):
  113. dburi = check_db_version(config.get_value("Database", "dburi"))
  114. return dburi
  115. def calculate_client_info(): # pragma: no cover
  116. import hashlib
  117. import socket
  118. from getpass import getuser
  119. hashinfo = hashlib.sha256((socket.gethostname() + getuser()).encode())
  120. hashinfo = hashinfo.hexdigest()
  121. return hashinfo
  122. def is_latest_version(version, client_info): # pragma: no cover
  123. """check current version againt latest version"""
  124. try:
  125. url = ("https://pwman.tiram.it/is_latest/?"
  126. "current_version={}&os={}&hash={}".format(
  127. version, sys.platform, client_info))
  128. res = urllib.request.urlopen(url, timeout=0.5)
  129. data = res.read() # This will return entire content.
  130. if res.status != 200:
  131. return None, True
  132. if data.decode().split(".") > version.split("."):
  133. return None, False
  134. else:
  135. return None, True
  136. except Exception as E:
  137. return E, True