mirror of
https://github.com/denoland/deno.git
synced 2024-12-22 07:14:47 -05:00
4d08bb85a4
* Make sync_third_party work in general * Un-break build.py and run_hooks.py on windows * Partially fix format.py on windows * Reduce code duplication between run_hooks and sync_third_party
91 lines
2.9 KiB
Python
91 lines
2.9 KiB
Python
# Copyright 2018 Ryan Dahl <ry@tinyclouds.org>
|
|
# All rights reserved. MIT License.
|
|
import os
|
|
import shutil
|
|
import stat
|
|
import subprocess
|
|
|
|
executable_suffix = ".exe" if os.name == "nt" else ""
|
|
root_path = os.path.dirname(os.path.dirname(os.path.realpath(__file__)))
|
|
|
|
|
|
def make_env(merge_env={}, env=None):
|
|
if env is None:
|
|
env = os.environ
|
|
env = env.copy()
|
|
for key in merge_env.keys():
|
|
env[key] = merge_env[key]
|
|
return env
|
|
|
|
|
|
def run(args, quiet=False, cwd=None, env=None, merge_env={}):
|
|
args[0] = os.path.normpath(args[0])
|
|
if not quiet:
|
|
print " ".join(args)
|
|
env = make_env(env=env, merge_env=merge_env)
|
|
shell = os.name == "nt" # Run through shell to make .bat/.cmd files work.
|
|
subprocess.check_call(args, cwd=cwd, env=env, shell=shell)
|
|
|
|
|
|
def remove_and_symlink(target, name, target_is_dir=False):
|
|
try:
|
|
# On Windows, directory symlink can only be removed with rmdir().
|
|
if os.name == "nt" and os.path.isdir(name):
|
|
os.rmdir(name)
|
|
else:
|
|
os.unlink(name)
|
|
except:
|
|
pass
|
|
symlink(target, name, target_is_dir)
|
|
|
|
|
|
def symlink(target, name, target_is_dir=False):
|
|
if os.name == "nt":
|
|
import ctypes
|
|
CreateSymbolicLinkW = ctypes.windll.kernel32.CreateSymbolicLinkW
|
|
CreateSymbolicLinkW.restype = ctypes.c_ubyte
|
|
CreateSymbolicLinkW.argtypes = (ctypes.c_wchar_p, ctypes.c_wchar_p,
|
|
ctypes.c_uint32)
|
|
|
|
# Replace forward slashes by backward slashes.
|
|
# Strangely it seems that this is only necessary for symlinks to files.
|
|
# Forward slashes don't cause any issues when the target is a directory.
|
|
target = target.replace("/", "\\")
|
|
flags = 0x02 # SYMBOLIC_LINK_FLAG_ALLOW_UNPRIVILEGED_CREATE
|
|
if (target_is_dir):
|
|
flags |= 0x01 # SYMBOLIC_LINK_FLAG_DIRECTORY
|
|
if not CreateSymbolicLinkW(name, target, flags):
|
|
raise ctypes.WinError()
|
|
else:
|
|
os.symlink(target, name)
|
|
|
|
|
|
def touch(fname):
|
|
if os.path.exists(fname):
|
|
os.utime(fname, None)
|
|
else:
|
|
open(fname, 'a').close()
|
|
|
|
|
|
# Recursive search for files of certain extensions.
|
|
# (Recursive glob doesn't exist in python 2.7.)
|
|
def find_exts(directory, *extensions):
|
|
matches = []
|
|
for root, dirnames, filenames in os.walk(directory):
|
|
for filename in filenames:
|
|
for ext in extensions:
|
|
if filename.endswith(ext):
|
|
matches.append(os.path.join(root, filename))
|
|
break
|
|
return matches
|
|
|
|
|
|
# The Python equivalent of `rm -rf`.
|
|
def rmtree(directory):
|
|
# On Windows, shutil.rmtree() won't delete files that have a readonly bit.
|
|
# Git creates some files that do. The 'onerror' callback deals with those.
|
|
def rm_readonly(func, path, _):
|
|
os.chmod(path, stat.S_IWRITE)
|
|
func(path)
|
|
|
|
shutil.rmtree(directory, onerror=rm_readonly)
|