-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathAdbPaste.py
executable file
·156 lines (124 loc) · 4.59 KB
/
AdbPaste.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
#!/usr/bin/python
#
# Copyright 2013 https://github.com/gcb
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
import sys, subprocess, re
class AdbPaste:
"Pass a long string as input to an android device/emulator"
def send(self, string, device=False, dryrun=False):
"encodes and sends a string to the device/emulator"
# The adb shell input command interprets "%s" as a
# space. Since there is no way to escape the % (e.g. %%
# is not treated specially), we split the string into
# multiple parts so each % character ends up at the end
# of a string where it will be left untouched. This also
# splits on % signs not part of a %s, in case any
# additional escape sequences are added to adb shell
# input later.
# See also https://github.com/gcb/AdbPaste/issues/1
for part in re.findall("[^%]+%?", string):
encoded = (
"$'" + "".join(["\\x" + c.encode("utf-8").hex() for c in part]) + "'"
)
self.sendEncoded(encoded, device, dryrun)
def sendEncoded(self, string, device, dryrun):
"sends a string to the device/emulator"
print(("sending", string))
if dryrun:
return
cmd = ["adb"]
if isinstance(device, str):
cmd += ["-s", device]
cmd += ["shell", "input", "text", string]
ret = subprocess.call(cmd)
if ret != 0:
if isinstance(ret, int):
sys.exit(ret)
else:
sys.exit(1)
def readFrom(f):
res = f.read()
if notab:
import re
res = re.sub("\t", " ", res)
return res
def displayHelp():
print(
"""
Command: python AdbPaste.py [options [optionArguments]] [text]
Options:
--help,-h: Show this help
--notab: Changes tabs into single spaces
-s: Serial number of the device adb should use. Analogous to -s flag of adb (use adb devices to see a list). Only needed if more than one device are available.
-n: Dry run. Echo what the command would do, without actually sending anything via adb.
--file: Next argument must be a filename. Content will be sent.
If --file is not used, and no text argument is specified, text is read from stdin.
"""
)
if __name__ == "__main__":
device = False
notab = False
arg_notab = "--notab"
arg_s = "-s"
arg_dryrun = "-n"
arg_file = "--file"
arg_help = "--help"
arg_help_short = "-h"
invalidArgMsg = (
"Invalid %s parameter. Run AdbPaste without any arguments to see help menu."
)
arg = sys.argv[1:]
# // --help,-h : Show help.
if arg_help in arg or arg_help_short in arg:
displayHelp()
sys.exit(1)
# // -n : Dry-run. Will not call adb, just echo out what it is doing.
if arg_dryrun in arg:
index = arg.index(arg_dryrun)
arg.pop(index)
dryrun = True
else:
dryrun = False
# // --notab: Convert tabs into spaces. usefull for 'typing' a file into a textarea or field where tab would change focus
if arg_notab in arg:
index = arg.index(arg_notab)
arg.pop(index)
notab = True
# // -s: pass the next value to -s flag on the adb command. For selecting which device to use when there's more than one present
if arg_s in arg:
index = arg.index(arg_s)
arg.pop(index) # Removes -s
if len(arg) == index:
print((invalidArgMsg % arg_s))
else:
device = arg[index]
arg.pop(index) # Removes device
# // -- file: read the contents from a file, and not from stdin
if arg_file in arg:
index = arg.index(arg_file)
arg.pop(index) # Remove --file
if len(arg) == index:
print((invalidArgMsg % arg_file))
elif isinstance(arg[index], str):
with open(arg[index], "r") as content_file:
arg = readFrom(content_file)
else:
arg = " ".join(arg)
else:
arg = " ".join(arg)
if not arg:
arg = readFrom(sys.stdin)
paste = AdbPaste()
paste.send(arg, device, dryrun)