mirror of
https://github.com/golang/go
synced 2024-11-22 00:04:41 -07:00
codereview: more utf-8 nonsense
To annoy hg: Μπορώ να φάω σπασμένα γυαλιά χωρίς να πάθω τίποτα. R=r CC=golang-dev https://golang.org/cl/2900041
This commit is contained in:
parent
d2d650bdc9
commit
3b6ddd9bce
@ -1,3 +1,7 @@
|
|||||||
|
# coding=utf-8
|
||||||
|
# (The line above is necessary so that I can use 世界 in the
|
||||||
|
# *comment* below without Python getting all bent out of shape.)
|
||||||
|
|
||||||
# Copyright 2007-2009 Google Inc.
|
# Copyright 2007-2009 Google Inc.
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
@ -108,6 +112,34 @@ defaultcc = None
|
|||||||
contributors = {}
|
contributors = {}
|
||||||
missing_codereview = None
|
missing_codereview = None
|
||||||
|
|
||||||
|
#######################################################################
|
||||||
|
# RE: UNICODE STRING HANDLING
|
||||||
|
#
|
||||||
|
# Python distinguishes between the str (string of bytes)
|
||||||
|
# and unicode (string of code points) types. Most operations
|
||||||
|
# work on either one just fine, but some (like regexp matching)
|
||||||
|
# require unicode, and others (like write) require str.
|
||||||
|
#
|
||||||
|
# As befits the language, Python hides the distinction between
|
||||||
|
# unicode and str by converting between them silently, but
|
||||||
|
# *only* if all the bytes/code points involved are 7-bit ASCII.
|
||||||
|
# This means that if you're not careful, your program works
|
||||||
|
# fine on "hello, world" and fails on "hello, 世界". And of course,
|
||||||
|
# the obvious way to be careful - use static types - is unavailable.
|
||||||
|
# So the only way is trial and error to find where to put explicit
|
||||||
|
# conversions.
|
||||||
|
#
|
||||||
|
# Because more functions do implicit conversion to str (string of bytes)
|
||||||
|
# than do implicit conversion to unicode (string of code points),
|
||||||
|
# the convention in this module is to represent all text as str,
|
||||||
|
# converting to unicode only when calling a unicode-only function
|
||||||
|
# and then converting back to str as soon as possible.
|
||||||
|
|
||||||
|
def typecheck(s, t):
|
||||||
|
if type(s) != t:
|
||||||
|
raise util.Abort("type check failed: %s has type %s != %s" % (repr(s), type(s), t))
|
||||||
|
|
||||||
|
|
||||||
#######################################################################
|
#######################################################################
|
||||||
# Change list parsing.
|
# Change list parsing.
|
||||||
#
|
#
|
||||||
@ -125,9 +157,9 @@ diff --git a/~rietveld~placeholder~ b/~rietveld~placeholder~
|
|||||||
new file mode 100644
|
new file mode 100644
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
class CL(object):
|
class CL(object):
|
||||||
def __init__(self, name):
|
def __init__(self, name):
|
||||||
|
typecheck(name, str)
|
||||||
self.name = name
|
self.name = name
|
||||||
self.desc = ''
|
self.desc = ''
|
||||||
self.files = []
|
self.files = []
|
||||||
@ -150,6 +182,7 @@ class CL(object):
|
|||||||
s += "Files:\n"
|
s += "Files:\n"
|
||||||
for f in cl.files:
|
for f in cl.files:
|
||||||
s += "\t" + f + "\n"
|
s += "\t" + f + "\n"
|
||||||
|
typecheck(s, str)
|
||||||
return s
|
return s
|
||||||
|
|
||||||
def EditorText(self):
|
def EditorText(self):
|
||||||
@ -174,6 +207,7 @@ class CL(object):
|
|||||||
for f in cl.files:
|
for f in cl.files:
|
||||||
s += "\t" + f + "\n"
|
s += "\t" + f + "\n"
|
||||||
s += "\n"
|
s += "\n"
|
||||||
|
typecheck(s, str)
|
||||||
return s
|
return s
|
||||||
|
|
||||||
def PendingText(self):
|
def PendingText(self):
|
||||||
@ -188,6 +222,7 @@ class CL(object):
|
|||||||
s += "\tFiles:\n"
|
s += "\tFiles:\n"
|
||||||
for f in cl.files:
|
for f in cl.files:
|
||||||
s += "\t\t" + f + "\n"
|
s += "\t\t" + f + "\n"
|
||||||
|
typecheck(s, str)
|
||||||
return s
|
return s
|
||||||
|
|
||||||
def Flush(self, ui, repo):
|
def Flush(self, ui, repo):
|
||||||
@ -215,6 +250,7 @@ class CL(object):
|
|||||||
s = s[0:55] + "..."
|
s = s[0:55] + "..."
|
||||||
if self.name != "new":
|
if self.name != "new":
|
||||||
s = "code review %s: %s" % (self.name, s)
|
s = "code review %s: %s" % (self.name, s)
|
||||||
|
typecheck(s, str)
|
||||||
return s
|
return s
|
||||||
|
|
||||||
def Upload(self, ui, repo, send_mail=False, gofmt=True, gofmt_just_warn=False):
|
def Upload(self, ui, repo, send_mail=False, gofmt=True, gofmt_just_warn=False):
|
||||||
@ -290,14 +326,18 @@ class CL(object):
|
|||||||
pmsg += "I'd like you to review this change.\n"
|
pmsg += "I'd like you to review this change.\n"
|
||||||
else:
|
else:
|
||||||
pmsg += "Please take another look.\n"
|
pmsg += "Please take another look.\n"
|
||||||
|
typecheck(pmsg, str)
|
||||||
PostMessage(ui, self.name, pmsg, subject=self.Subject())
|
PostMessage(ui, self.name, pmsg, subject=self.Subject())
|
||||||
self.mailed = True
|
self.mailed = True
|
||||||
self.Flush(ui, repo)
|
self.Flush(ui, repo)
|
||||||
|
|
||||||
def GoodCLName(name):
|
def GoodCLName(name):
|
||||||
|
typecheck(name, str)
|
||||||
return re.match("^[0-9]+$", name)
|
return re.match("^[0-9]+$", name)
|
||||||
|
|
||||||
def ParseCL(text, name):
|
def ParseCL(text, name):
|
||||||
|
typecheck(text, str)
|
||||||
|
typecheck(name, str)
|
||||||
sname = None
|
sname = None
|
||||||
lineno = 0
|
lineno = 0
|
||||||
sections = {
|
sections = {
|
||||||
@ -359,18 +399,22 @@ def ParseCL(text, name):
|
|||||||
return cl, 0, ''
|
return cl, 0, ''
|
||||||
|
|
||||||
def SplitCommaSpace(s):
|
def SplitCommaSpace(s):
|
||||||
|
typecheck(s, str)
|
||||||
s = s.strip()
|
s = s.strip()
|
||||||
if s == "":
|
if s == "":
|
||||||
return []
|
return []
|
||||||
return re.split(", *", s)
|
return re.split(", *", s)
|
||||||
|
|
||||||
def CutDomain(s):
|
def CutDomain(s):
|
||||||
|
typecheck(s, str)
|
||||||
i = s.find('@')
|
i = s.find('@')
|
||||||
if i >= 0:
|
if i >= 0:
|
||||||
s = s[0:i]
|
s = s[0:i]
|
||||||
return s
|
return s
|
||||||
|
|
||||||
def JoinComma(l):
|
def JoinComma(l):
|
||||||
|
for s in l:
|
||||||
|
typecheck(s, str)
|
||||||
return ", ".join(l)
|
return ", ".join(l)
|
||||||
|
|
||||||
def ExceptionDetail():
|
def ExceptionDetail():
|
||||||
@ -389,6 +433,7 @@ def IsLocalCL(ui, repo, name):
|
|||||||
|
|
||||||
# Load CL from disk and/or the web.
|
# Load CL from disk and/or the web.
|
||||||
def LoadCL(ui, repo, name, web=True):
|
def LoadCL(ui, repo, name, web=True):
|
||||||
|
typecheck(name, str)
|
||||||
set_status("loading CL " + name)
|
set_status("loading CL " + name)
|
||||||
if not GoodCLName(name):
|
if not GoodCLName(name):
|
||||||
return None, "invalid CL name"
|
return None, "invalid CL name"
|
||||||
@ -510,6 +555,7 @@ def RepoDir(ui, repo):
|
|||||||
url = url[5:]
|
url = url[5:]
|
||||||
if url.endswith('/'):
|
if url.endswith('/'):
|
||||||
url = url[:-1]
|
url = url[:-1]
|
||||||
|
typecheck(url, str)
|
||||||
return url
|
return url
|
||||||
|
|
||||||
# Find (or make) code review directory. On error, ui.warn and return None
|
# Find (or make) code review directory. On error, ui.warn and return None
|
||||||
@ -524,10 +570,12 @@ def CodeReviewDir(ui, repo):
|
|||||||
except:
|
except:
|
||||||
ui.warn('cannot mkdir %s: %s\n' % (dir, ExceptionDetail()))
|
ui.warn('cannot mkdir %s: %s\n' % (dir, ExceptionDetail()))
|
||||||
return None
|
return None
|
||||||
|
typecheck(dir, str)
|
||||||
return dir
|
return dir
|
||||||
|
|
||||||
# Strip maximal common leading white space prefix from text
|
# Strip maximal common leading white space prefix from text
|
||||||
def StripCommon(text):
|
def StripCommon(text):
|
||||||
|
typecheck(text, str)
|
||||||
ws = None
|
ws = None
|
||||||
for line in text.split('\n'):
|
for line in text.split('\n'):
|
||||||
line = line.rstrip()
|
line = line.rstrip()
|
||||||
@ -556,17 +604,22 @@ def StripCommon(text):
|
|||||||
t += line + '\n'
|
t += line + '\n'
|
||||||
while len(t) >= 2 and t[-2:] == '\n\n':
|
while len(t) >= 2 and t[-2:] == '\n\n':
|
||||||
t = t[:-1]
|
t = t[:-1]
|
||||||
|
typecheck(t, str)
|
||||||
return t
|
return t
|
||||||
|
|
||||||
# Indent text with indent.
|
# Indent text with indent.
|
||||||
def Indent(text, indent):
|
def Indent(text, indent):
|
||||||
|
typecheck(text, str)
|
||||||
|
typecheck(indent, str)
|
||||||
t = ''
|
t = ''
|
||||||
for line in text.split('\n'):
|
for line in text.split('\n'):
|
||||||
t += indent + line + '\n'
|
t += indent + line + '\n'
|
||||||
|
typecheck(t, str)
|
||||||
return t
|
return t
|
||||||
|
|
||||||
# Return the first line of l
|
# Return the first line of l
|
||||||
def line1(text):
|
def line1(text):
|
||||||
|
typecheck(text, str)
|
||||||
return text.split('\n')[0]
|
return text.split('\n')[0]
|
||||||
|
|
||||||
_change_prolog = """# Change list.
|
_change_prolog = """# Change list.
|
||||||
@ -1492,7 +1545,7 @@ class FormParser(HTMLParser):
|
|||||||
self.handle_data("&" + name + ";")
|
self.handle_data("&" + name + ";")
|
||||||
def handle_data(self, data):
|
def handle_data(self, data):
|
||||||
if self.curdata is not None:
|
if self.curdata is not None:
|
||||||
self.curdata += data.decode("utf-8").encode("utf-8")
|
self.curdata += data
|
||||||
|
|
||||||
# XML parser
|
# XML parser
|
||||||
def XMLGet(ui, path):
|
def XMLGet(ui, path):
|
||||||
@ -1632,6 +1685,10 @@ def MySend1(request_path, payload=None,
|
|||||||
f.close()
|
f.close()
|
||||||
# Translate \r\n into \n, because Rietveld doesn't.
|
# Translate \r\n into \n, because Rietveld doesn't.
|
||||||
response = response.replace('\r\n', '\n')
|
response = response.replace('\r\n', '\n')
|
||||||
|
# who knows what urllib will give us
|
||||||
|
if type(response) == unicode:
|
||||||
|
response = response.encode("utf-8")
|
||||||
|
typecheck(response, str)
|
||||||
return response
|
return response
|
||||||
except urllib2.HTTPError, e:
|
except urllib2.HTTPError, e:
|
||||||
if tries > 3:
|
if tries > 3:
|
||||||
@ -1650,11 +1707,13 @@ def MySend1(request_path, payload=None,
|
|||||||
|
|
||||||
def GetForm(url):
|
def GetForm(url):
|
||||||
f = FormParser()
|
f = FormParser()
|
||||||
f.feed(MySend(url))
|
f.feed(MySend(url).decode("utf-8")) # f.feed wants unicode
|
||||||
f.close()
|
f.close()
|
||||||
|
# convert back to utf-8 to restore sanity
|
||||||
|
m = {}
|
||||||
for k,v in f.map.items():
|
for k,v in f.map.items():
|
||||||
f.map[k] = v.replace("\r\n", "\n");
|
m[k.encode("utf-8")] = v.replace("\r\n", "\n").encode("utf-8")
|
||||||
return f.map
|
return m
|
||||||
|
|
||||||
# Fetch the settings for the CL, like reviewer and CC list, by
|
# Fetch the settings for the CL, like reviewer and CC list, by
|
||||||
# scraping the Rietveld editing forms.
|
# scraping the Rietveld editing forms.
|
||||||
@ -2265,17 +2324,16 @@ def EncodeMultipartFormData(fields, files):
|
|||||||
CRLF = '\r\n'
|
CRLF = '\r\n'
|
||||||
lines = []
|
lines = []
|
||||||
for (key, value) in fields:
|
for (key, value) in fields:
|
||||||
|
typecheck(key, str)
|
||||||
|
typecheck(value, str)
|
||||||
lines.append('--' + BOUNDARY)
|
lines.append('--' + BOUNDARY)
|
||||||
lines.append('Content-Disposition: form-data; name="%s"' % key)
|
lines.append('Content-Disposition: form-data; name="%s"' % key)
|
||||||
lines.append('')
|
lines.append('')
|
||||||
if type(value) == unicode:
|
|
||||||
value = value.encode("utf-8")
|
|
||||||
lines.append(value)
|
lines.append(value)
|
||||||
for (key, filename, value) in files:
|
for (key, filename, value) in files:
|
||||||
if type(filename) == unicode:
|
typecheck(key, str)
|
||||||
filename = filename.encode("utf-8")
|
typecheck(filename, str)
|
||||||
if type(value) == unicode:
|
typecheck(value, str)
|
||||||
value = value.encode("utf-8")
|
|
||||||
lines.append('--' + BOUNDARY)
|
lines.append('--' + BOUNDARY)
|
||||||
lines.append('Content-Disposition: form-data; name="%s"; filename="%s"' % (key, filename))
|
lines.append('Content-Disposition: form-data; name="%s"; filename="%s"' % (key, filename))
|
||||||
lines.append('Content-Type: %s' % GetContentType(filename))
|
lines.append('Content-Type: %s' % GetContentType(filename))
|
||||||
|
Loading…
Reference in New Issue
Block a user