2016-04-20 10:50:47 -05:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2019-07-24 11:49:25 -05:00
|
|
|
import os
|
2016-04-20 10:50:47 -05:00
|
|
|
import socket
|
|
|
|
import struct
|
|
|
|
import sys
|
|
|
|
|
|
|
|
# see com.intellij.idea.SocketLock for the server side of this interface
|
|
|
|
|
|
|
|
RUN_PATH = u'/opt/pycharm-professional/bin/pycharm.sh'
|
2019-07-24 11:49:25 -05:00
|
|
|
CONFIG_PATH = u'~/.PyCharm2019.2/config'
|
|
|
|
SYSTEM_PATH = u'~/.PyCharm2019.2/system'
|
2016-11-23 11:49:18 -05:00
|
|
|
|
2017-06-15 19:18:01 -05:00
|
|
|
|
2016-11-23 11:49:18 -05:00
|
|
|
def print_usage(cmd):
|
|
|
|
print(('Usage:\n' +
|
|
|
|
' {0} -h | -? | --help\n' +
|
2019-07-24 11:49:25 -05:00
|
|
|
' {0} [project_dir] [-w|--wait]\n' +
|
|
|
|
' {0} [-l|--line line] [project_dir|--temp-project] [-w|--wait] file[:line]\n' +
|
2016-11-23 11:49:18 -05:00
|
|
|
' {0} diff <left> <right>\n' +
|
|
|
|
' {0} merge <local> <remote> [base] <merged>').format(cmd))
|
|
|
|
|
|
|
|
|
2019-07-24 11:49:25 -05:00
|
|
|
def write_to_sock(sock, str):
|
|
|
|
if sys.version_info[0] >= 3: str = str.encode('utf-8')
|
|
|
|
sock.send(struct.pack('>h', len(str)) + str)
|
|
|
|
|
|
|
|
|
|
|
|
def read_from_sock(sock):
|
|
|
|
len = struct.unpack('>h', sock.recv(2))[0]
|
|
|
|
return sock.recv(len).decode('utf-8')
|
|
|
|
|
|
|
|
|
|
|
|
def read_sequence_from_sock(sock):
|
|
|
|
result = []
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
str = read_from_sock(sock)
|
|
|
|
if str == '---':
|
|
|
|
break
|
|
|
|
result.append(str)
|
|
|
|
|
|
|
|
except (socket.error, IOError) as e:
|
|
|
|
print("I/O error({0}): {1} ({2})".format(e.errno, e.strerror, e))
|
|
|
|
traceback.print_exception(*sys.exc_info())
|
|
|
|
return result
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2016-11-23 11:49:18 -05:00
|
|
|
def process_args(argv):
|
|
|
|
args = []
|
|
|
|
|
|
|
|
skip_next = False
|
|
|
|
for i, arg in enumerate(argv[1:]):
|
|
|
|
if arg == '-h' or arg == '-?' or arg == '--help':
|
|
|
|
print_usage(argv[0])
|
|
|
|
exit(0)
|
2017-06-15 19:18:01 -05:00
|
|
|
elif i == 0 and (arg == 'diff' or arg == 'merge' or arg == '--temp-project'):
|
2016-11-23 11:49:18 -05:00
|
|
|
args.append(arg)
|
|
|
|
elif arg == '-l' or arg == '--line':
|
|
|
|
args.append(arg)
|
|
|
|
skip_next = True
|
2019-07-24 11:49:25 -05:00
|
|
|
elif arg == '-w' or arg == '--wait':
|
|
|
|
args.append('--wait')
|
2016-11-23 11:49:18 -05:00
|
|
|
elif skip_next:
|
|
|
|
args.append(arg)
|
|
|
|
skip_next = False
|
|
|
|
else:
|
2017-03-24 14:18:44 -05:00
|
|
|
path = arg
|
2016-11-23 11:49:18 -05:00
|
|
|
if ':' in arg:
|
|
|
|
file_path, line_number = arg.rsplit(':', 1)
|
|
|
|
if line_number.isdigit():
|
|
|
|
args.append('-l')
|
|
|
|
args.append(line_number)
|
2017-03-24 14:18:44 -05:00
|
|
|
path = file_path
|
|
|
|
args.append(os.path.abspath(path))
|
2016-04-20 10:50:47 -05:00
|
|
|
|
2016-11-23 11:49:18 -05:00
|
|
|
return args
|
2016-04-20 10:50:47 -05:00
|
|
|
|
2016-11-23 11:49:18 -05:00
|
|
|
|
|
|
|
def try_activate_instance(args):
|
|
|
|
port_path = os.path.join(CONFIG_PATH, 'port')
|
|
|
|
token_path = os.path.join(SYSTEM_PATH, 'token')
|
|
|
|
if not (os.path.exists(port_path) and os.path.exists(token_path)):
|
|
|
|
return False
|
|
|
|
|
2018-07-25 19:14:04 -05:00
|
|
|
try:
|
|
|
|
with open(port_path) as pf:
|
|
|
|
port = int(pf.read())
|
|
|
|
with open(token_path) as tf:
|
|
|
|
token = tf.read()
|
|
|
|
except (ValueError):
|
|
|
|
return False
|
2016-04-20 10:50:47 -05:00
|
|
|
|
|
|
|
s = socket.socket()
|
2019-07-24 11:49:25 -05:00
|
|
|
s.settimeout(1.0)
|
2016-04-20 10:50:47 -05:00
|
|
|
try:
|
|
|
|
s.connect(('127.0.0.1', port))
|
2016-11-23 11:49:18 -05:00
|
|
|
except (socket.error, IOError):
|
2016-04-20 10:50:47 -05:00
|
|
|
return False
|
|
|
|
|
2019-07-24 11:49:25 -05:00
|
|
|
paths = read_sequence_from_sock(s)
|
|
|
|
found = CONFIG_PATH in paths
|
2016-04-20 10:50:47 -05:00
|
|
|
|
|
|
|
if found:
|
2019-07-24 11:49:25 -05:00
|
|
|
write_to_sock(s, 'activate ' + token + '\0' + os.getcwd() + '\0' + '\0'.join(args))
|
|
|
|
|
|
|
|
s.settimeout(None)
|
|
|
|
response = read_sequence_from_sock(s)
|
|
|
|
if response[0] != 'ok':
|
|
|
|
print('bad response: ' + response)
|
|
|
|
exit(1)
|
|
|
|
|
|
|
|
if len(response) > 2:
|
|
|
|
print(response[2])
|
|
|
|
|
|
|
|
exit(int(response[1]))
|
2016-04-20 10:50:47 -05:00
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2016-11-23 11:49:18 -05:00
|
|
|
def start_new_instance(args):
|
|
|
|
if sys.platform == 'darwin':
|
|
|
|
if len(args) > 0:
|
|
|
|
args.insert(0, '--args')
|
2018-07-25 19:14:04 -05:00
|
|
|
os.execvp('/usr/bin/open', ['-a', RUN_PATH] + args)
|
2016-05-11 10:11:57 -05:00
|
|
|
else:
|
2016-11-23 11:49:18 -05:00
|
|
|
bin_file = os.path.split(RUN_PATH)[1]
|
2016-05-11 10:11:57 -05:00
|
|
|
os.execv(RUN_PATH, [bin_file] + args)
|
2016-11-23 11:49:18 -05:00
|
|
|
|
|
|
|
|
|
|
|
ide_args = process_args(sys.argv)
|
|
|
|
if not try_activate_instance(ide_args):
|
|
|
|
start_new_instance(ide_args)
|