123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343 |
- from __future__ import print_function
- import sys
- import os
- import io
- import getopt
- import re
- import string
- import errno
- import copy
- from jsbeautifier.__version__ import __version__
- #
- # The MIT License (MIT)
- # Copyright (c) 2007-2017 Einar Lielmanis, Liam Newman, and contributors.
- # Permission is hereby granted, free of charge, to any person
- # obtaining a copy of this software and associated documentation files
- # (the "Software"), to deal in the Software without restriction,
- # including without limitation the rights to use, copy, modify, merge,
- # publish, distribute, sublicense, and/or sell copies of the Software,
- # and to permit persons to whom the Software is furnished to do so,
- # subject to the following conditions:
- # The above copyright notice and this permission notice shall be
- # included in all copies or substantial portions of the Software.
- # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
- # BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
- # ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
- # CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- # SOFTWARE.
- #
- # Originally written by Einar Lielmanis et al.,
- # Conversion to python by Einar Lielmanis, einar@jsbeautifier.org,
- # Parsing improvement for brace-less and semicolon-less statements
- # by Liam Newman <bitwiseman@gmail.com>
- # Python is not my native language, feel free to push things around.
- #
- # Use either from command line (script displays its usage when run
- # without any parameters),
- #
- #
- # or, alternatively, use it as a module:
- #
- # import jsbeautifier
- # res = jsbeautifier.beautify('your javascript string')
- # res = jsbeautifier.beautify_file('some_file.js')
- #
- # you may specify some options:
- #
- # opts = jsbeautifier.default_options()
- # opts.indent_size = 2
- # res = jsbeautifier.beautify('some javascript', opts)
- #
- #
- # Here are the available options: (read source)
- class BeautifierOptions:
- def __init__(self):
- self.indent_size = 4
- self.indent_char = ' '
- self.indent_with_tabs = False
- self.eol = 'auto'
- self.preserve_newlines = True
- self.max_preserve_newlines = 10
- self.space_in_paren = False
- self.space_in_empty_paren = False
- self.e4x = False
- self.jslint_happy = False
- self.space_after_anon_function = False
- self.brace_style = 'collapse'
- self.keep_array_indentation = False
- self.keep_function_indentation = False
- self.eval_code = False
- self.unescape_strings = False
- self.wrap_line_length = 0
- self.break_chained_methods = False
- self.end_with_newline = False
- self.comma_first = False
- self.operator_position = 'before-newline'
- self.css = None
- self.js = None
- self.html = None
- # For testing of beautify ignore:start directive
- self.test_output_raw = False
- self.editorconfig = False
- def mergeOpts(self, targetType):
- finalOpts = copy.copy(self)
- local = getattr(finalOpts, targetType)
- if (local):
- delattr(finalOpts, targetType)
- for key in local:
- setattr(finalOpts, key, local[key])
- return finalOpts
- def __repr__(self):
- return \
- """indent_size = %d
- indent_char = [%s]
- preserve_newlines = %s
- max_preserve_newlines = %d
- space_in_paren = %s
- jslint_happy = %s
- space_after_anon_function = %s
- indent_with_tabs = %s
- brace_style = %s
- keep_array_indentation = %s
- eval_code = %s
- wrap_line_length = %s
- unescape_strings = %s
- """ % ( self.indent_size,
- self.indent_char,
- self.preserve_newlines,
- self.max_preserve_newlines,
- self.space_in_paren,
- self.jslint_happy,
- self.space_after_anon_function,
- self.indent_with_tabs,
- self.brace_style,
- self.keep_array_indentation,
- self.eval_code,
- self.wrap_line_length,
- self.unescape_strings,
- )
- class BeautifierFlags:
- def __init__(self, mode):
- self.mode = mode
- self.parent = None
- self.last_text = ''
- self.last_word = ''
- self.declaration_statement = False
- self.declaration_assignment = False
- self.multiline_frame = False
- self.inline_frame = False
- self.if_block = False
- self.else_block = False
- self.do_block = False
- self.do_while = False
- self.import_block = False
- self.in_case = False
- self.in_case_statement = False
- self.case_body = False
- self.indentation_level = 0
- self.line_indent_level = 0
- self.start_line_index = 0
- self.ternary_depth = 0
- def apply_base(self, flags_base, added_newline):
- next_indent_level = flags_base.indentation_level
- if not added_newline and \
- flags_base.line_indent_level > next_indent_level:
- next_indent_level = flags_base.line_indent_level
- self.parent = flags_base
- self.last_text = flags_base.last_text
- self.last_word = flags_base.last_word
- self.indentation_level = next_indent_level
- class Acorn:
- def __init__(self):
- # This is not pretty, but given how we did the version import
- # it is the only way to do this without having setup.py fail on a missing six dependency.
- self.six = __import__("six")
- # This section of code was translated to python from acorn (javascript).
- #
- # Acorn was written by Marijn Haverbeke and released under an MIT
- # license. The Unicode regexps (for identifiers and whitespace) were
- # taken from [Esprima](http://esprima.org) by Ariya Hidayat.
- #
- # Git repositories for Acorn are available at
- #
- # http://marijnhaverbeke.nl/git/acorn
- # https://github.com/marijnh/acorn.git
- # ## Character categories
- # Big ugly regular expressions that match characters in the
- # whitespace, identifier, and identifier-start categories. These
- # are only applied when a character is found to actually have a
- # code point above 128.
- self.nonASCIIwhitespace = re.compile(self.six.u("[\u1680\u180e\u2000-\u200a\u202f\u205f\u3000\ufeff]"))
- self.nonASCIIidentifierStartChars = self.six.u("\xaa\xb5\xba\xc0-\xd6\xd8-\xf6\xf8-\u02c1\u02c6-\u02d1\u02e0-\u02e4\u02ec\u02ee\u0370-\u0374\u0376\u0377\u037a-\u037d\u0386\u0388-\u038a\u038c\u038e-\u03a1\u03a3-\u03f5\u03f7-\u0481\u048a-\u0527\u0531-\u0556\u0559\u0561-\u0587\u05d0-\u05ea\u05f0-\u05f2\u0620-\u064a\u066e\u066f\u0671-\u06d3\u06d5\u06e5\u06e6\u06ee\u06ef\u06fa-\u06fc\u06ff\u0710\u0712-\u072f\u074d-\u07a5\u07b1\u07ca-\u07ea\u07f4\u07f5\u07fa\u0800-\u0815\u081a\u0824\u0828\u0840-\u0858\u08a0\u08a2-\u08ac\u0904-\u0939\u093d\u0950\u0958-\u0961\u0971-\u0977\u0979-\u097f\u0985-\u098c\u098f\u0990\u0993-\u09a8\u09aa-\u09b0\u09b2\u09b6-\u09b9\u09bd\u09ce\u09dc\u09dd\u09df-\u09e1\u09f0\u09f1\u0a05-\u0a0a\u0a0f\u0a10\u0a13-\u0a28\u0a2a-\u0a30\u0a32\u0a33\u0a35\u0a36\u0a38\u0a39\u0a59-\u0a5c\u0a5e\u0a72-\u0a74\u0a85-\u0a8d\u0a8f-\u0a91\u0a93-\u0aa8\u0aaa-\u0ab0\u0ab2\u0ab3\u0ab5-\u0ab9\u0abd\u0ad0\u0ae0\u0ae1\u0b05-\u0b0c\u0b0f\u0b10\u0b13-\u0b28\u0b2a-\u0b30\u0b32\u0b33\u0b35-\u0b39\u0b3d\u0b5c\u0b5d\u0b5f-\u0b61\u0b71\u0b83\u0b85-\u0b8a\u0b8e-\u0b90\u0b92-\u0b95\u0b99\u0b9a\u0b9c\u0b9e\u0b9f\u0ba3\u0ba4\u0ba8-\u0baa\u0bae-\u0bb9\u0bd0\u0c05-\u0c0c\u0c0e-\u0c10\u0c12-\u0c28\u0c2a-\u0c33\u0c35-\u0c39\u0c3d\u0c58\u0c59\u0c60\u0c61\u0c85-\u0c8c\u0c8e-\u0c90\u0c92-\u0ca8\u0caa-\u0cb3\u0cb5-\u0cb9\u0cbd\u0cde\u0ce0\u0ce1\u0cf1\u0cf2\u0d05-\u0d0c\u0d0e-\u0d10\u0d12-\u0d3a\u0d3d\u0d4e\u0d60\u0d61\u0d7a-\u0d7f\u0d85-\u0d96\u0d9a-\u0db1\u0db3-\u0dbb\u0dbd\u0dc0-\u0dc6\u0e01-\u0e30\u0e32\u0e33\u0e40-\u0e46\u0e81\u0e82\u0e84\u0e87\u0e88\u0e8a\u0e8d\u0e94-\u0e97\u0e99-\u0e9f\u0ea1-\u0ea3\u0ea5\u0ea7\u0eaa\u0eab\u0ead-\u0eb0\u0eb2\u0eb3\u0ebd\u0ec0-\u0ec4\u0ec6\u0edc-\u0edf\u0f00\u0f40-\u0f47\u0f49-\u0f6c\u0f88-\u0f8c\u1000-\u102a\u103f\u1050-\u1055\u105a-\u105d\u1061\u1065\u1066\u106e-\u1070\u1075-\u1081\u108e\u10a0-\u10c5\u10c7\u10cd\u10d0-\u10fa\u10fc-\u1248\u124a-\u124d\u1250-\u1256\u1258\u125a-\u125d\u1260-\u1288\u128a-\u128d\u1290-\u12b0\u12b2-\u12b5\u12b8-\u12be\u12c0\u12c2-\u12c5\u12c8-\u12d6\u12d8-\u1310\u1312-\u1315\u1318-\u135a\u1380-\u138f\u13a0-\u13f4\u1401-\u166c\u166f-\u167f\u1681-\u169a\u16a0-\u16ea\u16ee-\u16f0\u1700-\u170c\u170e-\u1711\u1720-\u1731\u1740-\u1751\u1760-\u176c\u176e-\u1770\u1780-\u17b3\u17d7\u17dc\u1820-\u1877\u1880-\u18a8\u18aa\u18b0-\u18f5\u1900-\u191c\u1950-\u196d\u1970-\u1974\u1980-\u19ab\u19c1-\u19c7\u1a00-\u1a16\u1a20-\u1a54\u1aa7\u1b05-\u1b33\u1b45-\u1b4b\u1b83-\u1ba0\u1bae\u1baf\u1bba-\u1be5\u1c00-\u1c23\u1c4d-\u1c4f\u1c5a-\u1c7d\u1ce9-\u1cec\u1cee-\u1cf1\u1cf5\u1cf6\u1d00-\u1dbf\u1e00-\u1f15\u1f18-\u1f1d\u1f20-\u1f45\u1f48-\u1f4d\u1f50-\u1f57\u1f59\u1f5b\u1f5d\u1f5f-\u1f7d\u1f80-\u1fb4\u1fb6-\u1fbc\u1fbe\u1fc2-\u1fc4\u1fc6-\u1fcc\u1fd0-\u1fd3\u1fd6-\u1fdb\u1fe0-\u1fec\u1ff2-\u1ff4\u1ff6-\u1ffc\u2071\u207f\u2090-\u209c\u2102\u2107\u210a-\u2113\u2115\u2119-\u211d\u2124\u2126\u2128\u212a-\u212d\u212f-\u2139\u213c-\u213f\u2145-\u2149\u214e\u2160-\u2188\u2c00-\u2c2e\u2c30-\u2c5e\u2c60-\u2ce4\u2ceb-\u2cee\u2cf2\u2cf3\u2d00-\u2d25\u2d27\u2d2d\u2d30-\u2d67\u2d6f\u2d80-\u2d96\u2da0-\u2da6\u2da8-\u2dae\u2db0-\u2db6\u2db8-\u2dbe\u2dc0-\u2dc6\u2dc8-\u2dce\u2dd0-\u2dd6\u2dd8-\u2dde\u2e2f\u3005-\u3007\u3021-\u3029\u3031-\u3035\u3038-\u303c\u3041-\u3096\u309d-\u309f\u30a1-\u30fa\u30fc-\u30ff\u3105-\u312d\u3131-\u318e\u31a0-\u31ba\u31f0-\u31ff\u3400-\u4db5\u4e00-\u9fcc\ua000-\ua48c\ua4d0-\ua4fd\ua500-\ua60c\ua610-\ua61f\ua62a\ua62b\ua640-\ua66e\ua67f-\ua697\ua6a0-\ua6ef\ua717-\ua71f\ua722-\ua788\ua78b-\ua78e\ua790-\ua793\ua7a0-\ua7aa\ua7f8-\ua801\ua803-\ua805\ua807-\ua80a\ua80c-\ua822\ua840-\ua873\ua882-\ua8b3\ua8f2-\ua8f7\ua8fb\ua90a-\ua925\ua930-\ua946\ua960-\ua97c\ua984-\ua9b2\ua9cf\uaa00-\uaa28\uaa40-\uaa42\uaa44-\uaa4b\uaa60-\uaa76\uaa7a\uaa80-\uaaaf\uaab1\uaab5\uaab6\uaab9-\uaabd\uaac0\uaac2\uaadb-\uaadd\uaae0-\uaaea\uaaf2-\uaaf4\uab01-\uab06\uab09-\uab0e\uab11-\uab16\uab20-\uab26\uab28-\uab2e\uabc0-\uabe2\uac00-\ud7a3\ud7b0-\ud7c6\ud7cb-\ud7fb\uf900-\ufa6d\ufa70-\ufad9\ufb00-\ufb06\ufb13-\ufb17\ufb1d\ufb1f-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb3e\ufb40\ufb41\ufb43\ufb44\ufb46-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe70-\ufe74\ufe76-\ufefc\uff21-\uff3a\uff41-\uff5a\uff66-\uffbe\uffc2-\uffc7\uffca-\uffcf\uffd2-\uffd7\uffda-\uffdc")
- self.nonASCIIidentifierChars = self.six.u("\u0300-\u036f\u0483-\u0487\u0591-\u05bd\u05bf\u05c1\u05c2\u05c4\u05c5\u05c7\u0610-\u061a\u0620-\u0649\u0672-\u06d3\u06e7-\u06e8\u06fb-\u06fc\u0730-\u074a\u0800-\u0814\u081b-\u0823\u0825-\u0827\u0829-\u082d\u0840-\u0857\u08e4-\u08fe\u0900-\u0903\u093a-\u093c\u093e-\u094f\u0951-\u0957\u0962-\u0963\u0966-\u096f\u0981-\u0983\u09bc\u09be-\u09c4\u09c7\u09c8\u09d7\u09df-\u09e0\u0a01-\u0a03\u0a3c\u0a3e-\u0a42\u0a47\u0a48\u0a4b-\u0a4d\u0a51\u0a66-\u0a71\u0a75\u0a81-\u0a83\u0abc\u0abe-\u0ac5\u0ac7-\u0ac9\u0acb-\u0acd\u0ae2-\u0ae3\u0ae6-\u0aef\u0b01-\u0b03\u0b3c\u0b3e-\u0b44\u0b47\u0b48\u0b4b-\u0b4d\u0b56\u0b57\u0b5f-\u0b60\u0b66-\u0b6f\u0b82\u0bbe-\u0bc2\u0bc6-\u0bc8\u0bca-\u0bcd\u0bd7\u0be6-\u0bef\u0c01-\u0c03\u0c46-\u0c48\u0c4a-\u0c4d\u0c55\u0c56\u0c62-\u0c63\u0c66-\u0c6f\u0c82\u0c83\u0cbc\u0cbe-\u0cc4\u0cc6-\u0cc8\u0cca-\u0ccd\u0cd5\u0cd6\u0ce2-\u0ce3\u0ce6-\u0cef\u0d02\u0d03\u0d46-\u0d48\u0d57\u0d62-\u0d63\u0d66-\u0d6f\u0d82\u0d83\u0dca\u0dcf-\u0dd4\u0dd6\u0dd8-\u0ddf\u0df2\u0df3\u0e34-\u0e3a\u0e40-\u0e45\u0e50-\u0e59\u0eb4-\u0eb9\u0ec8-\u0ecd\u0ed0-\u0ed9\u0f18\u0f19\u0f20-\u0f29\u0f35\u0f37\u0f39\u0f41-\u0f47\u0f71-\u0f84\u0f86-\u0f87\u0f8d-\u0f97\u0f99-\u0fbc\u0fc6\u1000-\u1029\u1040-\u1049\u1067-\u106d\u1071-\u1074\u1082-\u108d\u108f-\u109d\u135d-\u135f\u170e-\u1710\u1720-\u1730\u1740-\u1750\u1772\u1773\u1780-\u17b2\u17dd\u17e0-\u17e9\u180b-\u180d\u1810-\u1819\u1920-\u192b\u1930-\u193b\u1951-\u196d\u19b0-\u19c0\u19c8-\u19c9\u19d0-\u19d9\u1a00-\u1a15\u1a20-\u1a53\u1a60-\u1a7c\u1a7f-\u1a89\u1a90-\u1a99\u1b46-\u1b4b\u1b50-\u1b59\u1b6b-\u1b73\u1bb0-\u1bb9\u1be6-\u1bf3\u1c00-\u1c22\u1c40-\u1c49\u1c5b-\u1c7d\u1cd0-\u1cd2\u1d00-\u1dbe\u1e01-\u1f15\u200c\u200d\u203f\u2040\u2054\u20d0-\u20dc\u20e1\u20e5-\u20f0\u2d81-\u2d96\u2de0-\u2dff\u3021-\u3028\u3099\u309a\ua640-\ua66d\ua674-\ua67d\ua69f\ua6f0-\ua6f1\ua7f8-\ua800\ua806\ua80b\ua823-\ua827\ua880-\ua881\ua8b4-\ua8c4\ua8d0-\ua8d9\ua8f3-\ua8f7\ua900-\ua909\ua926-\ua92d\ua930-\ua945\ua980-\ua983\ua9b3-\ua9c0\uaa00-\uaa27\uaa40-\uaa41\uaa4c-\uaa4d\uaa50-\uaa59\uaa7b\uaae0-\uaae9\uaaf2-\uaaf3\uabc0-\uabe1\uabec\uabed\uabf0-\uabf9\ufb20-\ufb28\ufe00-\ufe0f\ufe20-\ufe26\ufe33\ufe34\ufe4d-\ufe4f\uff10-\uff19\uff3f")
- self.nonASCIIidentifierStart = re.compile("[" + self.nonASCIIidentifierStartChars + "]")
- self.nonASCIIidentifier = re.compile("[" + self.nonASCIIidentifierStartChars + self.nonASCIIidentifierChars + "]")
- # Whether a single character denotes a newline.
- self.newline = re.compile(self.six.u("[\n\r\u2028\u2029]"))
- # Matches a whole line break (where CRLF is considered a single
- # line break). Used to count lines.
- # in javascript, these two differ
- # in python they are the same, different methods are called on them
- self.lineBreak = re.compile(self.six.u("\r\n|[\n\r\u2028\u2029]"))
- self.allLineBreaks = self.lineBreak
- # Test whether a given character code starts an identifier.
- def isIdentifierStart(self, code):
- if code < 65:
- return code in [36, 64] # permit $ (36) and @ (64). @ is used in ES7 decorators.
- if code < 91:
- return True # 65 through 91 are uppercase letters
- if code < 97:
- return code == 95 # permit _ (95)
- if code < 123:
- return True # 97 through 123 are lowercase letters
- return code >= 0xaa and self.nonASCIIidentifierStart.match(self.six.unichr(code)) != None
- # Test whether a given character is part of an identifier.
- def isIdentifierChar(self, code):
- if code < 48:
- return code == 36
- if code < 58:
- return True
- if code < 65:
- return False
- if code < 91:
- return True
- if code < 97:
- return code == 95
- if code < 123:
- return True
- return code >= 0xaa and self.nonASCIIidentifier.match(self.six.unichr(code)) != None
- class Token:
- def __init__(self, type, text, newlines = 0, whitespace_before = '', mode = None, parent = None):
- self.type = type
- self.text = text
- self.comments_before = []
- self.newlines = newlines
- self.wanted_newline = newlines > 0
- self.whitespace_before = whitespace_before
- self.parent = None
- self.opened = None
- self.directives = None
- def default_options():
- return BeautifierOptions()
- def beautify(string, opts = default_options() ):
- b = Beautifier()
- return b.beautify(string, opts)
- def set_file_editorconfig_opts(filename, js_options):
- from editorconfig import get_properties, EditorConfigError
- try:
- _ecoptions = get_properties(os.path.abspath(filename))
- if _ecoptions.get("indent_style") == "tab":
- js_options.indent_with_tabs = True
- elif _ecoptions.get("indent_style") == "space":
- js_options.indent_with_tabs = False
- if _ecoptions.get("indent_size"):
- js_options.indent_size = int(_ecoptions["indent_size"])
- if _ecoptions.get("max_line_length"):
- if _ecoptions.get("max_line_length") == "off":
- js_options.wrap_line_length = 0
- else:
- js_options.wrap_line_length = int(_ecoptions["max_line_length"])
- if _ecoptions.get("insert_final_newline") == 'true':
- js_options.end_with_newline = True
- elif _ecoptions.get("insert_final_newline") == 'false':
- js_options.end_with_newline = False
- if _ecoptions.get("end_of_line"):
- if _ecoptions["end_of_line"] == "cr":
- js_options.eol = '\r'
- elif _ecoptions["end_of_line"] == "lf":
- js_options.eol = '\n'
- elif _ecoptions["end_of_line"] == "crlf":
- js_options.eol = '\r\n'
- except EditorConfigError as ex:
- # do not error on bad editor config
- print("Error loading EditorConfig. Ignoring.", file=sys.stderr)
- def beautify_file(file_name, opts = default_options() ):
- input_string = ''
- if file_name == '-': # stdin
- try:
- if sys.stdin.isatty():
- raise Exception()
- stream = sys.stdin
- input_string = ''.join(stream.readlines())
- except Exception as ex:
- print("Must pipe input or define at least one file.", file=sys.stderr)
- usage(sys.stderr)
- raise Exception()
- else:
- stream = io.open(file_name, 'rt', newline='')
- input_string = ''.join(stream.readlines())
- return beautify(input_string, opts)
- def usage(stream=sys.stdout):
- print("jsbeautifier.py@" + __version__ + """
- Javascript beautifier (http://jsbeautifier.org/)
- Usage: jsbeautifier.py [options] <infile>
- <infile> can be "-", which means stdin.
- <outfile> defaults to stdout
- Input options:
- -i, --stdin Read input from stdin
- Output options:
- -s, --indent-size=NUMBER Indentation size. (default 4).
- -c, --indent-char=CHAR Character to indent with. (default space).
- -e, --eol=STRING Character(s) to use as line terminators.
- (default first newline in file, otherwise "\\n")
- -t, --indent-with-tabs Indent with tabs, overrides -s and -c
- -d, --disable-preserve-newlines Do not preserve existing line breaks.
- -P, --space-in-paren Add padding spaces within paren, ie. f( a, b )
- -E, --space-in-empty-paren Add a single space inside empty paren, ie. f( )
- -j, --jslint-happy More jslint-compatible output
- -a, --space_after_anon_function Add a space before an anonymous function's parens, ie. function ()
- -b, --brace-style=collapse Brace style (collapse, expand, end-expand, none)(,preserve-inline)
- -k, --keep-array-indentation Keep array indentation.
- -r, --replace Write output in-place, replacing input
- -o, --outfile=FILE Specify a file to output to (default stdout)
- -f, --keep-function-indentation Do not re-indent function bodies defined in var lines.
- -x, --unescape-strings Decode printable chars encoded in \\xNN notation.
- -X, --e4x Pass E4X xml literals through untouched
- -w, --wrap-line-length Attempt to wrap line when it exceeds this length.
- NOTE: Line continues until next wrap point is found.
- -n, --end_with_newline End output with newline
- --editorconfig Enable setting configuration from EditorConfig
- Rarely needed options:
- --eval-code evaluate code if a JS interpreter is
- installed. May be useful with some obfuscated
- script but poses a potential security issue.
- -l, --indent-level=NUMBER Initial indentation level. (default 0).
- -h, --help, --usage Prints this help statement.
- -v, --version Show the version
- """, file=stream)
- if stream == sys.stderr:
- return 1
- else:
- return 0
- OPERATOR_POSITION = {
- 'before_newline': 'before-newline',
- 'after_newline': 'after-newline',
- 'preserve_newline': 'preserve-newline'
- }
- OPERATOR_POSITION_BEFORE_OR_PRESERVE = [OPERATOR_POSITION['before_newline'], OPERATOR_POSITION['preserve_newline']];
- def sanitizeOperatorPosition(opPosition):
- if not opPosition:
- return OPERATOR_POSITION['before_newline']
- elif opPosition not in OPERATOR_POSITION.values():
- raise ValueError("Invalid Option Value: The option 'operator_position' must be one of the following values\n" +
- str(OPERATOR_POSITION.values()) +
- "\nYou passed in: '" + opPosition + "'")
- return opPosition
- class MODE:
- BlockStatement, Statement, ObjectLiteral, ArrayLiteral, \
- ForInitializer, Conditional, Expression = range(7)
- class Beautifier:
- def __init__(self, opts = default_options() ):
- self.opts = copy.copy(opts)
- self.acorn = Acorn()
- self.blank_state()
- def blank_state(self, js_source_text = None):
- # internal flags
- self.flags = None
- self.previous_flags = None
- self.flag_store = []
- self.tokens = []
- self.token_pos = 0
- # force opts.space_after_anon_function to true if opts.jslint_happy
- if self.opts.jslint_happy:
- self.opts.space_after_anon_function = True
- if self.opts.indent_with_tabs:
- self.opts.indent_char = "\t"
- self.opts.indent_size = 1
- if self.opts.eol == 'auto':
- self.opts.eol = '\n'
- if self.acorn.lineBreak.search(js_source_text or ''):
- self.opts.eol = self.acorn.lineBreak.search(js_source_text).group()
- self.opts.eol = self.opts.eol.replace('\\r', '\r').replace('\\n', '\n')
- self.indent_string = self.opts.indent_char * self.opts.indent_size
- self.baseIndentString = ''
- self.last_type = 'TK_START_BLOCK' # last token type
- self.last_last_text = '' # pre-last token text
- preindent_index = 0;
- if not js_source_text == None and len(js_source_text) > 0:
- while preindent_index < len(js_source_text) and \
- js_source_text[preindent_index] in [' ', '\t'] :
- self.baseIndentString += js_source_text[preindent_index]
- preindent_index += 1
- js_source_text = js_source_text[preindent_index:]
- self.output = Output(self.indent_string, self.baseIndentString)
- # If testing the ignore directive, start with output disable set to true
- self.output.raw = self.opts.test_output_raw;
- self.set_mode(MODE.BlockStatement)
- return js_source_text
- def beautify(self, s, opts = None ):
- if opts != None:
- opts = opts.mergeOpts('js')
- self.opts = copy.copy(opts)
- #Compat with old form
- if self.opts.brace_style == 'collapse-preserve-inline':
- self.opts.brace_style = 'collapse,preserve-inline'
- split = re.compile("[^a-zA-Z0-9_\-]+").split(self.opts.brace_style)
- self.opts.brace_style = split[0]
- self.opts.brace_preserve_inline = (True if bool(split[1] == 'preserve-inline') else None) if len(split) > 1 else False
- if self.opts.brace_style not in ['expand', 'collapse', 'end-expand', 'none']:
- raise(Exception('opts.brace_style must be "expand", "collapse", "end-expand", or "none".'))
- if self.opts.brace_preserve_inline == None:
- raise(Exception('opts.brace_style second item must be "preserve-inline"'))
- s = self.blank_state(s)
- input = self.unpack(s, self.opts.eval_code)
- self.handlers = {
- 'TK_START_EXPR': self.handle_start_expr,
- 'TK_END_EXPR': self.handle_end_expr,
- 'TK_START_BLOCK': self.handle_start_block,
- 'TK_END_BLOCK': self.handle_end_block,
- 'TK_WORD': self.handle_word,
- 'TK_RESERVED': self.handle_word,
- 'TK_SEMICOLON': self.handle_semicolon,
- 'TK_STRING': self.handle_string,
- 'TK_EQUALS': self.handle_equals,
- 'TK_OPERATOR': self.handle_operator,
- 'TK_COMMA': self.handle_comma,
- 'TK_BLOCK_COMMENT': self.handle_block_comment,
- 'TK_COMMENT': self.handle_comment,
- 'TK_DOT': self.handle_dot,
- 'TK_UNKNOWN': self.handle_unknown,
- 'TK_EOF': self.handle_eof
- }
- self.tokens = Tokenizer(input, self.opts, self.indent_string).tokenize()
- self.token_pos = 0
- current_token = self.get_token()
- while current_token != None:
- self.handlers[current_token.type](current_token)
- self.last_last_text = self.flags.last_text
- self.last_type = current_token.type
- self.flags.last_text = current_token.text
- self.token_pos += 1
- current_token = self.get_token()
- sweet_code = self.output.get_code()
- if self.opts.end_with_newline:
- sweet_code += '\n'
- if not self.opts.eol == '\n':
- sweet_code = sweet_code.replace('\n', self.opts.eol)
- return sweet_code
- def handle_whitespace_and_comments(self, local_token, preserve_statement_flags = False):
- newlines = local_token.newlines
- keep_whitespace = self.opts.keep_array_indentation and self.is_array(self.flags.mode)
- for comment_token in local_token.comments_before:
- # The cleanest handling of inline comments is to treat them as though they aren't there.
- # Just continue formatting and the behavior should be logical.
- # Also ignore unknown tokens. Again, this should result in better behavior.
- self.handle_whitespace_and_comments(comment_token, preserve_statement_flags)
- self.handlers[comment_token.type](comment_token, preserve_statement_flags)
- if keep_whitespace:
- for i in range(newlines):
- self.print_newline(i > 0, preserve_statement_flags)
- else: # not keep_whitespace
- if self.opts.max_preserve_newlines != 0 and newlines > self.opts.max_preserve_newlines:
- newlines = self.opts.max_preserve_newlines
- if self.opts.preserve_newlines and newlines > 1:
- self.print_newline(False, preserve_statement_flags)
- for i in range(1, newlines):
- self.print_newline(True, preserve_statement_flags)
- def unpack(self, source, evalcode=False):
- import jsbeautifier.unpackers as unpackers
- try:
- return unpackers.run(source, evalcode)
- except unpackers.UnpackingError as error:
- return source
- def is_special_word(self, s):
- return s in ['case', 'return', 'do', 'if', 'throw', 'else']
- def is_array(self, mode):
- return mode == MODE.ArrayLiteral
- def is_expression(self, mode):
- return mode in [MODE.Expression, MODE.ForInitializer, MODE.Conditional]
- _newline_restricted_tokens = ['break','continue','return', 'throw']
- def allow_wrap_or_preserved_newline(self, current_token, force_linewrap = False):
- # never wrap the first token of a line.
- if self.output.just_added_newline():
- return
- shouldPreserveOrForce = (self.opts.preserve_newlines and current_token.wanted_newline) or force_linewrap
- operatorLogicApplies = self.flags.last_text in Tokenizer.positionable_operators or current_token.text in Tokenizer.positionable_operators
- if operatorLogicApplies:
- shouldPrintOperatorNewline = (self.flags.last_text in Tokenizer.positionable_operators and self.opts.operator_position in OPERATOR_POSITION_BEFORE_OR_PRESERVE) \
- or current_token.text in Tokenizer.positionable_operators
- shouldPreserveOrForce = shouldPreserveOrForce and shouldPrintOperatorNewline
- if shouldPreserveOrForce:
- self.print_newline(preserve_statement_flags = True)
- elif self.opts.wrap_line_length > 0:
- if self.last_type == 'TK_RESERVED' and self.flags.last_text in self._newline_restricted_tokens:
- # These tokens should never have a newline inserted between
- # them and the following expression.
- return
- proposed_line_length = self.output.current_line.get_character_count() + len(current_token.text)
- if self.output.space_before_token:
- proposed_line_length += 1
- if proposed_line_length >= self.opts.wrap_line_length:
- self.print_newline(preserve_statement_flags = True)
- def print_newline(self, force_newline = False, preserve_statement_flags = False):
- if not preserve_statement_flags:
- if self.flags.last_text != ';' and self.flags.last_text != ',' and self.flags.last_text != '=' and self.last_type != 'TK_OPERATOR':
- next_token = self.get_token(1)
- while (self.flags.mode == MODE.Statement and
- not (self.flags.if_block and next_token and next_token.type == 'TK_RESERVED' and next_token.text == 'else') and
- not self.flags.do_block):
- self.restore_mode()
- if self.output.add_new_line(force_newline):
- self.flags.multiline_frame = True
- def print_token_line_indentation(self, current_token):
- if self.output.just_added_newline():
- line = self.output.current_line
- if self.opts.keep_array_indentation and self.is_array(self.flags.mode) and current_token.wanted_newline:
- line.push(current_token.whitespace_before)
- self.output.space_before_token = False
- elif self.output.set_indent(self.flags.indentation_level):
- self.flags.line_indent_level = self.flags.indentation_level
- def print_token(self, current_token, s=None):
- if self.output.raw:
- self.output.add_raw_token(current_token)
- return
- if self.opts.comma_first and self.last_type == 'TK_COMMA' and self.output.just_added_newline():
- if self.output.previous_line.last() == ',':
- # if the comma was already at the start of the line,
- # pull back onto that line and reprint the indentation
- popped = self.output.previous_line.pop()
- if self.output.previous_line.is_empty():
- self.output.previous_line.push(popped)
- self.output.trim(True)
- self.output.current_line.pop()
- self.output.trim()
- # add the comma in front of the next token
- self.print_token_line_indentation(current_token)
- self.output.add_token(',')
- self.output.space_before_token = True
- if s == None:
- s = current_token.text
- self.print_token_line_indentation(current_token)
- self.output.add_token(s);
- def indent(self):
- self.flags.indentation_level += 1
- def deindent(self):
- allow_deindent = self.flags.indentation_level > 0 and ((self.flags.parent == None) or self.flags.indentation_level > self.flags.parent.indentation_level)
- if allow_deindent:
- self.flags.indentation_level -= 1
- def set_mode(self, mode):
- if self.flags:
- self.flag_store.append(self.flags)
- self.previous_flags = self.flags
- else:
- self.previous_flags = BeautifierFlags(mode)
- self.flags = BeautifierFlags(mode)
- self.flags.apply_base(self.previous_flags, self.output.just_added_newline())
- self.flags.start_line_index = self.output.get_line_number();
- def restore_mode(self):
- if len(self.flag_store) > 0:
- self.previous_flags = self.flags
- self.flags = self.flag_store.pop()
- if self.previous_flags.mode == MODE.Statement:
- self.output.remove_redundant_indentation(self.previous_flags)
- def start_of_object_property(self):
- return self.flags.parent.mode == MODE.ObjectLiteral and self.flags.mode == MODE.Statement and \
- ((self.flags.last_text == ':' and self.flags.ternary_depth == 0) or (self.last_type == 'TK_RESERVED' and self.flags.last_text in ['get', 'set']))
- def start_of_statement(self, current_token):
- if (
- (self.last_type == 'TK_RESERVED' and self.flags.last_text in ['var', 'let', 'const'] and current_token.type == 'TK_WORD') \
- or (self.last_type == 'TK_RESERVED' and self.flags.last_text== 'do') \
- or (self.last_type == 'TK_RESERVED' and self.flags.last_text in ['return', 'throw'] and not current_token.wanted_newline) \
- or (self.last_type == 'TK_RESERVED' and self.flags.last_text == 'else' \
- and not (current_token.type == 'TK_RESERVED' and current_token.text == 'if' and not len(current_token.comments_before))) \
- or (self.last_type == 'TK_END_EXPR' and (self.previous_flags.mode == MODE.ForInitializer or self.previous_flags.mode == MODE.Conditional)) \
- or (self.last_type == 'TK_WORD' and self.flags.mode == MODE.BlockStatement \
- and not self.flags.in_case
- and not (current_token.text == '--' or current_token.text == '++')
- and self.last_last_text != 'function'
- and current_token.type != 'TK_WORD' and current_token.type != 'TK_RESERVED') \
- or (self.flags.mode == MODE.ObjectLiteral and \
- ((self.flags.last_text == ':' and self.flags.ternary_depth == 0) or (self.last_type == 'TK_RESERVED' and self.flags.last_text in ['get', 'set'])))
- ):
- self.set_mode(MODE.Statement)
- self.indent()
- self.handle_whitespace_and_comments(current_token, True);
- # Issue #276:
- # If starting a new statement with [if, for, while, do], push to a new line.
- # if (a) if (b) if(c) d(); else e(); else f();
- if not self.start_of_object_property():
- self.allow_wrap_or_preserved_newline(current_token, current_token.type == 'TK_RESERVED' and current_token.text in ['do', 'for', 'if', 'while'])
- return True
- else:
- return False
- def get_token(self, offset = 0):
- index = self.token_pos + offset
- if index < 0 or index >= len(self.tokens):
- return None
- else:
- return self.tokens[index]
- def handle_start_expr(self, current_token):
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- pass
- else:
- self.handle_whitespace_and_comments(current_token)
- next_mode = MODE.Expression
- if current_token.text == '[':
- if self.last_type == 'TK_WORD' or self.flags.last_text == ')':
- if self.last_type == 'TK_RESERVED' and self.flags.last_text in Tokenizer.line_starters:
- self.output.space_before_token = True
- self.set_mode(next_mode)
- self.print_token(current_token)
- self.indent()
- if self.opts.space_in_paren:
- self.output.space_before_token = True
- return
- next_mode = MODE.ArrayLiteral
- if self.is_array(self.flags.mode):
- if self.flags.last_text == '[' or (
- self.flags.last_text == ',' and (self.last_last_text == ']' or self.last_last_text == '}')):
- # ], [ goes to a new line
- # }, [ goes to a new line
- if not self.opts.keep_array_indentation:
- self.print_newline()
- else:
- if self.last_type == 'TK_RESERVED' and self.flags.last_text == 'for':
- next_mode = MODE.ForInitializer
- elif self.last_type == 'TK_RESERVED' and self.flags.last_text in ['if', 'while']:
- next_mode = MODE.Conditional
- else:
- next_mode = MODE.Expression
- if self.flags.last_text == ';' or self.last_type == 'TK_START_BLOCK':
- self.print_newline()
- elif self.last_type in ['TK_END_EXPR', 'TK_START_EXPR', 'TK_END_BLOCK'] or self.flags.last_text == '.':
- # do nothing on (( and )( and ][ and ]( and .(
- # TODO: Consider whether forcing this is required. Review failing tests when removed.
- self.allow_wrap_or_preserved_newline(current_token, current_token.wanted_newline)
- elif not (self.last_type == 'TK_RESERVED' and current_token.text == '(') and self.last_type not in ['TK_WORD', 'TK_OPERATOR']:
- self.output.space_before_token = True
- elif (self.last_type == 'TK_RESERVED' and (self.flags.last_word == 'function' or self.flags.last_word == 'typeof')) or \
- (self.flags.last_text == '*' and (
- self.last_last_text in ['function', 'yield'] or
- (self.flags.mode == MODE.ObjectLiteral and self.last_last_text in ['{', ',']))):
- # function() vs function (), typeof() vs typeof ()
- # function*() vs function* (), yield*() vs yield* ()
- if self.opts.space_after_anon_function:
- self.output.space_before_token = True
- elif self.last_type == 'TK_RESERVED' and (self.flags.last_text in Tokenizer.line_starters or self.flags.last_text == 'catch'):
- # TODO: option space_before_conditional
- self.output.space_before_token = True
- elif current_token.text == '(' and self.last_type == 'TK_RESERVED' and self.flags.last_word == 'await':
- self.output.space_before_token = True
- # Support of this kind of newline preservation:
- # a = (b &&
- # (c || d));
- if self.last_type in ['TK_EQUALS', 'TK_OPERATOR']:
- if not self.start_of_object_property():
- self.allow_wrap_or_preserved_newline(current_token)
- # Support preserving wrapped arrow function expressions
- # a.b('c',
- # () => d.e
- # )
- if current_token.text == '(' and self.last_type not in ['TK_WORD', 'TK_RESERVED']:
- self.allow_wrap_or_preserved_newline(current_token)
- self.set_mode(next_mode)
- self.print_token(current_token)
- if self.opts.space_in_paren:
- self.output.space_before_token = True
- # In all cases, if we newline while inside an expression it should be indented.
- self.indent()
- def handle_end_expr(self, current_token):
- # statements inside expressions are not valid syntax, but...
- # statements must all be closed when their container closes
- while self.flags.mode == MODE.Statement:
- self.restore_mode()
- self.handle_whitespace_and_comments(current_token)
- if self.flags.multiline_frame:
- self.allow_wrap_or_preserved_newline(current_token, current_token.text == ']' and self.is_array(self.flags.mode) and not self.opts.keep_array_indentation)
- if self.opts.space_in_paren:
- if self.last_type == 'TK_START_EXPR' and not self.opts.space_in_empty_paren:
- # empty parens are always "()" and "[]", not "( )" or "[ ]"
- self.output.space_before_token = False
- self.output.trim()
- else:
- self.output.space_before_token = True
- if current_token.text == ']' and self.opts.keep_array_indentation:
- self.print_token(current_token)
- self.restore_mode()
- else:
- self.restore_mode()
- self.print_token(current_token)
- self.output.remove_redundant_indentation(self.previous_flags)
- # do {} while () // no statement required after
- if self.flags.do_while and self.previous_flags.mode == MODE.Conditional:
- self.previous_flags.mode = MODE.Expression
- self.flags.do_block = False
- self.flags.do_while = False
- def handle_start_block(self, current_token):
- self.handle_whitespace_and_comments(current_token)
- # Check if this is a BlockStatement that should be treated as a ObjectLiteral
- next_token = self.get_token(1)
- second_token = self.get_token(2)
- if second_token != None and \
- ((second_token.text in [':', ','] and next_token.type in ['TK_STRING', 'TK_WORD', 'TK_RESERVED']) \
- or (next_token.text in ['get', 'set', '...'] and second_token.type in ['TK_WORD', 'TK_RESERVED'])):
- # We don't support TypeScript,but we didn't break it for a very long time.
- # We'll try to keep not breaking it.
- if not self.last_last_text in ['class','interface']:
- self.set_mode(MODE.ObjectLiteral)
- else:
- self.set_mode(MODE.BlockStatement)
- elif self.last_type == 'TK_OPERATOR' and self.flags.last_text == '=>':
- # arrow function: (param1, paramN) => { statements }
- self.set_mode(MODE.BlockStatement)
- elif self.last_type in ['TK_EQUALS', 'TK_START_EXPR', 'TK_COMMA', 'TK_OPERATOR'] or \
- (self.last_type == 'TK_RESERVED' and self.flags.last_text in ['return', 'throw', 'import', 'default']):
- # Detecting shorthand function syntax is difficult by scanning forward,
- # so check the surrounding context.
- # If the block is being returned, imported, export default, passed as arg,
- # assigned with = or assigned in a nested object, treat as an ObjectLiteral.
- self.set_mode(MODE.ObjectLiteral)
- else:
- self.set_mode(MODE.BlockStatement)
- empty_braces = (not next_token == None) and len(next_token.comments_before) == 0 and next_token.text == '}'
- empty_anonymous_function = empty_braces and self.flags.last_word == 'function' and \
- self.last_type == 'TK_END_EXPR'
- if self.opts.brace_preserve_inline: # check for inline, set inline_frame if so
- # search forward for newline wanted inside this block
- index = 0
- check_token = None
- self.flags.inline_frame = True
- do_loop = True
- while (do_loop):
- index += 1
- check_token = self.get_token(index)
- if check_token.wanted_newline:
- self.flags.inline_frame = False
- do_loop = (check_token.type != 'TK_EOF' and
- not (check_token.type == 'TK_END_BLOCK' and check_token.opened == current_token))
- if (self.opts.brace_style == 'expand' or \
- (self.opts.brace_style == 'none' and current_token.wanted_newline)) and \
- not self.flags.inline_frame:
- if self.last_type != 'TK_OPERATOR' and \
- (empty_anonymous_function or
- self.last_type == 'TK_EQUALS' or
- (self.last_type == 'TK_RESERVED' and self.is_special_word(self.flags.last_text) and self.flags.last_text != 'else')):
- self.output.space_before_token = True
- else:
- self.print_newline(preserve_statement_flags = True)
- else: # collapse || inline_frame
- if self.is_array(self.previous_flags.mode) and (self.last_type == 'TK_START_EXPR' or self.last_type == 'TK_COMMA'):
- # if we're preserving inline,
- # allow newline between comma and next brace.
- if self.flags.inline_frame:
- self.allow_wrap_or_preserved_newline(current_token)
- self.flags.inline_frame = True
- self.previous_flags.multiline_frame = self.previous_flags.multiline_frame or self.flags.multiline_frame
- self.flags.multiline_frame = False
- elif self.last_type == 'TK_COMMA':
- self.output.space_before_token = True
- elif self.last_type not in ['TK_OPERATOR', 'TK_START_EXPR']:
- if self.last_type == 'TK_START_BLOCK' and not self.flags.inline_frame:
- self.print_newline()
- else:
- self.output.space_before_token = True
- self.print_token(current_token)
- self.indent()
- def handle_end_block(self, current_token):
- # statements must all be closed when their container closes
- self.handle_whitespace_and_comments(current_token)
- while self.flags.mode == MODE.Statement:
- self.restore_mode()
- empty_braces = self.last_type == 'TK_START_BLOCK'
- if self.flags.inline_frame and not empty_braces: # try inline_frame (only set if opt.braces-preserve-inline) first
- self.output.space_before_token = True;
- elif self.opts.brace_style == 'expand':
- if not empty_braces:
- self.print_newline()
- else:
- # skip {}
- if not empty_braces:
- if self.is_array(self.flags.mode) and self.opts.keep_array_indentation:
- self.opts.keep_array_indentation = False
- self.print_newline()
- self.opts.keep_array_indentation = True
- else:
- self.print_newline()
- self.restore_mode()
- self.print_token(current_token)
- def handle_word(self, current_token):
- if current_token.type == 'TK_RESERVED':
- if current_token.text in ['set', 'get'] and self.flags.mode != MODE.ObjectLiteral:
- current_token.type = 'TK_WORD'
- elif current_token.text in ['as', 'from'] and not self.flags.import_block:
- current_token.type = 'TK_WORD'
- elif self.flags.mode == MODE.ObjectLiteral:
- next_token = self.get_token(1)
- if next_token.text == ':':
- current_token.type = 'TK_WORD'
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- if self.last_type == 'TK_RESERVED' and self.flags.last_text in ['var', 'let', 'const'] and current_token.type == 'TK_WORD':
- self.flags.declaration_statement = True
- elif current_token.wanted_newline and \
- not self.is_expression(self.flags.mode) and \
- (self.last_type != 'TK_OPERATOR' or (self.flags.last_text == '--' or self.flags.last_text == '++')) and \
- self.last_type != 'TK_EQUALS' and \
- (self.opts.preserve_newlines or not (self.last_type == 'TK_RESERVED' and self.flags.last_text in ['var', 'let', 'const', 'set', 'get'])):
- self.handle_whitespace_and_comments(current_token)
- self.print_newline()
- else:
- self.handle_whitespace_and_comments(current_token)
- if self.flags.do_block and not self.flags.do_while:
- if current_token.type == 'TK_RESERVED' and current_token.text == 'while':
- # do {} ## while ()
- self.output.space_before_token = True
- self.print_token(current_token)
- self.output.space_before_token = True
- self.flags.do_while = True
- return
- else:
- # do {} should always have while as the next word.
- # if we don't see the expected while, recover
- self.print_newline()
- self.flags.do_block = False
- # if may be followed by else, or not
- # Bare/inline ifs are tricky
- # Need to unwind the modes correctly: if (a) if (b) c(); else d(); else e();
- if self.flags.if_block:
- if (not self.flags.else_block) and (current_token.type == 'TK_RESERVED' and current_token.text == 'else'):
- self.flags.else_block = True
- else:
- while self.flags.mode == MODE.Statement:
- self.restore_mode()
- self.flags.if_block = False
- if current_token.type == 'TK_RESERVED' and (current_token.text == 'case' or (current_token.text == 'default' and self.flags.in_case_statement)):
- self.print_newline()
- if self.flags.case_body or self.opts.jslint_happy:
- self.flags.case_body = False
- self.deindent()
- self.print_token(current_token)
- self.flags.in_case = True
- self.flags.in_case_statement = True
- return
- if self.last_type in ['TK_COMMA', 'TK_START_EXPR', 'TK_EQUALS', 'TK_OPERATOR']:
- if not self.start_of_object_property():
- self.allow_wrap_or_preserved_newline(current_token)
- if current_token.type == 'TK_RESERVED' and current_token.text == 'function':
- if (self.flags.last_text in ['}', ';'] or
- (self.output.just_added_newline() and not (self.flags.last_text in ['(', '[', '{', ':', '=', ','] or self.last_type == 'TK_OPERATOR'))):
- # make sure there is a nice clean space of at least one blank line
- # before a new function definition, except in arrays
- if not self.output.just_added_blankline() and len(current_token.comments_before) == 0:
- self.print_newline()
- self.print_newline(True)
- if self.last_type == 'TK_RESERVED' or self.last_type == 'TK_WORD':
- if self.last_type == 'TK_RESERVED' and self.flags.last_text in ['get', 'set', 'new', 'return', 'export', 'async']:
- self.output.space_before_token = True
- elif self.last_type == 'TK_RESERVED' and self.flags.last_text == 'default' and self.last_last_text == 'export':
- self.output.space_before_token = True
- else:
- self.print_newline()
- elif self.last_type == 'TK_OPERATOR' or self.flags.last_text == '=':
- # foo = function
- self.output.space_before_token = True
- elif not self.flags.multiline_frame and (self.is_expression(self.flags.mode) or self.is_array(self.flags.mode)):
- # (function
- pass
- else:
- self.print_newline()
- self.print_token(current_token)
- self.flags.last_word = current_token.text
- return
- prefix = 'NONE'
- if self.last_type == 'TK_END_BLOCK':
- if self.previous_flags.inline_frame:
- prefix = 'SPACE'
- elif not (current_token.type == 'TK_RESERVED' and current_token.text in ['else', 'catch', 'finally', 'from']):
- prefix = 'NEWLINE'
- else:
- if self.opts.brace_style in ['expand', 'end-expand'] or \
- (self.opts.brace_style == 'none' and current_token.wanted_newline):
- prefix = 'NEWLINE'
- else:
- prefix = 'SPACE'
- self.output.space_before_token = True
- elif self.last_type == 'TK_SEMICOLON' and self.flags.mode == MODE.BlockStatement:
- # TODO: Should this be for STATEMENT as well?
- prefix = 'NEWLINE'
- elif self.last_type == 'TK_SEMICOLON' and self.is_expression(self.flags.mode):
- prefix = 'SPACE'
- elif self.last_type == 'TK_STRING':
- prefix = 'NEWLINE'
- elif self.last_type == 'TK_RESERVED' or self.last_type == 'TK_WORD' or \
- (self.flags.last_text == '*' and (
- self.last_last_text in ['function', 'yield'] or
- (self.flags.mode == MODE.ObjectLiteral and self.last_last_text in ['{', ',']))):
- prefix = 'SPACE'
- elif self.last_type == 'TK_START_BLOCK':
- if self.flags.inline_frame:
- prefix = 'SPACE'
- else:
- prefix = 'NEWLINE'
- elif self.last_type == 'TK_END_EXPR':
- self.output.space_before_token = True
- prefix = 'NEWLINE'
- if current_token.type == 'TK_RESERVED' and current_token.text in Tokenizer.line_starters and self.flags.last_text != ')':
- if self.flags.inline_frame or self.flags.last_text == 'else ' or self.flags.last_text == 'export':
- prefix = 'SPACE'
- else:
- prefix = 'NEWLINE'
- if current_token.type == 'TK_RESERVED' and current_token.text in ['else', 'catch', 'finally']:
- if ((not (self.last_type == 'TK_END_BLOCK' and self.previous_flags.mode == MODE.BlockStatement)) \
- or self.opts.brace_style == 'expand' \
- or self.opts.brace_style == 'end-expand' \
- or (self.opts.brace_style == 'none' and current_token.wanted_newline)) \
- and not self.flags.inline_frame:
- self.print_newline()
- else:
- self.output.trim(True)
- # If we trimmed and there's something other than a close block before us
- # put a newline back in. Handles '} // comment' scenario.
- if self.output.current_line.last() != '}':
- self.print_newline()
- self.output.space_before_token = True
- elif prefix == 'NEWLINE':
- if self.last_type == 'TK_RESERVED' and self.is_special_word(self.flags.last_text):
- # no newline between return nnn
- self.output.space_before_token = True
- elif self.last_type != 'TK_END_EXPR':
- if (self.last_type != 'TK_START_EXPR' or not (current_token.type == 'TK_RESERVED' and current_token.text in ['var', 'let', 'const'])) and self.flags.last_text != ':':
- # no need to force newline on VAR -
- # for (var x = 0...
- if current_token.type == 'TK_RESERVED' and current_token.text == 'if' and self.flags.last_text == 'else':
- self.output.space_before_token = True
- else:
- self.print_newline()
- elif current_token.type == 'TK_RESERVED' and current_token.text in Tokenizer.line_starters and self.flags.last_text != ')':
- self.print_newline()
- elif self.flags.multiline_frame and self.is_array(self.flags.mode) and self.flags.last_text == ',' and self.last_last_text == '}':
- self.print_newline() # }, in lists get a newline
- elif prefix == 'SPACE':
- self.output.space_before_token = True
- self.print_token(current_token)
- self.flags.last_word = current_token.text
- if current_token.type == 'TK_RESERVED':
- if current_token.text == 'do':
- self.flags.do_block = True
- elif current_token.text == 'if':
- self.flags.if_block = True
- elif current_token.text == 'import':
- self.flags.import_block = True
- elif current_token.text == 'from' and self.flags.import_block:
- self.flags.import_block = False
- def handle_semicolon(self, current_token):
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- # Semicolon can be the start (and end) of a statement
- self.output.space_before_token = False
- else:
- self.handle_whitespace_and_comments(current_token)
- next_token = self.get_token(1)
- while (self.flags.mode == MODE.Statement and
- not (self.flags.if_block and next_token and next_token.type == 'TK_RESERVED' and next_token.text == 'else') and
- not self.flags.do_block):
- self.restore_mode()
- if self.flags.import_block:
- self.flags.import_block = False
- self.print_token(current_token)
- def handle_string(self, current_token):
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- # One difference - strings want at least a space before
- self.output.space_before_token = True
- else:
- self.handle_whitespace_and_comments(current_token)
- if self.last_type == 'TK_RESERVED' or self.last_type == 'TK_WORD' or self.flags.inline_frame:
- self.output.space_before_token = True
- elif self.last_type in ['TK_COMMA', 'TK_START_EXPR', 'TK_EQUALS', 'TK_OPERATOR']:
- if not self.start_of_object_property():
- self.allow_wrap_or_preserved_newline(current_token)
- else:
- self.print_newline()
- self.print_token(current_token)
- def handle_equals(self, current_token):
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- pass
- else:
- self.handle_whitespace_and_comments(current_token)
- if self.flags.declaration_statement:
- # just got an '=' in a var-line, different line breaking rules will apply
- self.flags.declaration_assignment = True
- self.output.space_before_token = True
- self.print_token(current_token)
- self.output.space_before_token = True
- def handle_comma(self, current_token):
- self.handle_whitespace_and_comments(current_token, True)
- self.print_token(current_token)
- self.output.space_before_token = True
- if self.flags.declaration_statement:
- if self.is_expression(self.flags.parent.mode):
- # do not break on comma, for ( var a = 1, b = 2
- self.flags.declaration_assignment = False
- if self.flags.declaration_assignment:
- self.flags.declaration_assignment = False
- self.print_newline(preserve_statement_flags = True)
- elif self.opts.comma_first:
- # for comma-first, we want to allow a newline before the comma
- # to turn into a newline after the comma, which we will fixup later
- self.allow_wrap_or_preserved_newline(current_token)
- elif self.flags.mode == MODE.ObjectLiteral \
- or (self.flags.mode == MODE.Statement and self.flags.parent.mode == MODE.ObjectLiteral):
- if self.flags.mode == MODE.Statement:
- self.restore_mode()
- if not self.flags.inline_frame:
- self.print_newline()
- elif self.opts.comma_first:
- # EXPR or DO_BLOCK
- # for comma-first, we want to allow a newline before the comma
- # to turn into a newline after the comma, which we will fixup later
- self.allow_wrap_or_preserved_newline(current_token)
- def handle_operator(self, current_token):
- isGeneratorAsterisk = current_token.text == '*' and \
- ((self.last_type == 'TK_RESERVED' and self.flags.last_text in ['function', 'yield']) or
- (self.last_type in ['TK_START_BLOCK', 'TK_COMMA', 'TK_END_BLOCK', 'TK_SEMICOLON']))
- isUnary = current_token.text in ['+', '-'] \
- and (self.last_type in ['TK_START_BLOCK', 'TK_START_EXPR', 'TK_EQUALS', 'TK_OPERATOR'] \
- or self.flags.last_text in Tokenizer.line_starters or self.flags.last_text == ',')
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- pass
- else:
- preserve_statement_flags = not isGeneratorAsterisk
- self.handle_whitespace_and_comments(current_token, preserve_statement_flags)
- if self.last_type == 'TK_RESERVED' and self.is_special_word(self.flags.last_text):
- # return had a special handling in TK_WORD
- self.output.space_before_token = True
- self.print_token(current_token)
- return
- # hack for actionscript's import .*;
- if current_token.text == '*' and self.last_type == 'TK_DOT':
- self.print_token(current_token)
- return
- if current_token.text == '::':
- # no spaces around the exotic namespacing syntax operator
- self.print_token(current_token)
- return
- # Allow line wrapping between operators when operator_position is
- # set to before or preserve
- if self.last_type == 'TK_OPERATOR' and self.opts.operator_position in OPERATOR_POSITION_BEFORE_OR_PRESERVE:
- self.allow_wrap_or_preserved_newline(current_token)
- if current_token.text == ':' and self.flags.in_case:
- self.flags.case_body = True
- self.indent()
- self.print_token(current_token)
- self.print_newline()
- self.flags.in_case = False
- return
- space_before = True
- space_after = True
- in_ternary = False
- if current_token.text == ':':
- if self.flags.ternary_depth == 0:
- # Colon is invalid javascript outside of ternary and object, but do our best to guess what was meant.
- space_before = False
- else:
- self.flags.ternary_depth -= 1
- in_ternary = True
- elif current_token.text == '?':
- self.flags.ternary_depth += 1
- # let's handle the operator_position option prior to any conflicting logic
- if (not isUnary) and (not isGeneratorAsterisk) and \
- self.opts.preserve_newlines and current_token.text in Tokenizer.positionable_operators:
- isColon = current_token.text == ':'
- isTernaryColon = isColon and in_ternary
- isOtherColon = isColon and not in_ternary
- if self.opts.operator_position == OPERATOR_POSITION['before_newline']:
- # if the current token is : and it's not a ternary statement then we set space_before to false
- self.output.space_before_token = not isOtherColon
- self.print_token(current_token)
- if (not isColon) or isTernaryColon:
- self.allow_wrap_or_preserved_newline(current_token)
- self.output.space_before_token = True
- return
- elif self.opts.operator_position == OPERATOR_POSITION['after_newline']:
- # if the current token is anything but colon, or (via deduction) it's a colon and in a ternary statement,
- # then print a newline.
- self.output.space_before_token = True
- if (not isColon) or isTernaryColon:
- if self.get_token(1).wanted_newline:
- self.print_newline(preserve_statement_flags = True)
- else:
- self.allow_wrap_or_preserved_newline(current_token)
- else:
- self.output.space_before_token = False
- self.print_token(current_token)
- self.output.space_before_token = True
- return
- elif self.opts.operator_position == OPERATOR_POSITION['preserve_newline']:
- if not isOtherColon:
- self.allow_wrap_or_preserved_newline(current_token)
- # if we just added a newline, or the current token is : and it's not a ternary statement,
- # then we set space_before to false
- self.output.space_before_token = not (self.output.just_added_newline() or isOtherColon)
- self.print_token(current_token)
- self.output.space_before_token = True
- return
- if isGeneratorAsterisk:
- self.allow_wrap_or_preserved_newline(current_token)
- space_before = False
- next_token = self.get_token(1)
- space_after = next_token and next_token.type in ['TK_WORD','TK_RESERVED']
- elif current_token.text == '...':
- self.allow_wrap_or_preserved_newline(current_token)
- space_before = self.last_type == 'TK_START_BLOCK'
- space_after = False
- elif current_token.text in ['--', '++', '!', '~'] or isUnary:
- space_before = False
- space_after = False
- # http://www.ecma-international.org/ecma-262/5.1/#sec-7.9.1
- # if there is a newline between -- or ++ and anything else we should preserve it.
- if current_token.wanted_newline and (current_token.text == '--' or current_token.text == '++'):
- self.print_newline(preserve_statement_flags = True)
- if self.flags.last_text == ';' and self.is_expression(self.flags.mode):
- # for (;; ++i)
- # ^^
- space_before = True
- if self.last_type == 'TK_RESERVED':
- space_before = True
- elif self.last_type == 'TK_END_EXPR':
- space_before = not (self.flags.last_text == ']' and current_token.text in ['--', '++'])
- elif self.last_type == 'TK_OPERATOR':
- # a++ + ++b
- # a - -b
- space_before = current_token.text in ['--', '-','++', '+'] and self.flags.last_text in ['--', '-','++', '+']
- # + and - are not unary when preceeded by -- or ++ operator
- # a-- + b
- # a * +b
- # a - -b
- if current_token.text in ['-', '+'] and self.flags.last_text in ['--', '++']:
- space_after = True
- if (((self.flags.mode == MODE.BlockStatement and not self.flags.inline_frame) or self.flags.mode == MODE.Statement)
- and self.flags.last_text in ['{', ';']):
- # { foo: --i }
- # foo(): --bar
- self.print_newline()
- if space_before:
- self.output.space_before_token = True
- self.print_token(current_token)
- if space_after:
- self.output.space_before_token = True
- def handle_block_comment(self, current_token, preserve_statement_flags):
- if self.output.raw:
- self.output.add_raw_token(current_token)
- if current_token.directives and current_token.directives.get('preserve') == 'end':
- # If we're testing the raw output behavior, do not allow a directive to turn it off.
- self.output.raw = self.opts.test_output_raw
- return
- if current_token.directives:
- self.print_newline(preserve_statement_flags = preserve_statement_flags)
- self.print_token(current_token)
- if current_token.directives.get('preserve') == 'start':
- self.output.raw = True
- self.print_newline(preserve_statement_flags = True)
- return
- # inline block
- if not self.acorn.newline.search(current_token.text) and not current_token.wanted_newline:
- self.output.space_before_token = True
- self.print_token(current_token)
- self.output.space_before_token = True
- return
- lines = self.acorn.allLineBreaks.split(current_token.text)
- javadoc = False
- starless = False
- last_indent = current_token.whitespace_before
- last_indent_length = len(last_indent)
- # block comment starts with a new line
- self.print_newline(preserve_statement_flags = preserve_statement_flags)
- if len(lines) > 1:
- javadoc = not any(l for l in lines[1:] if ( l.strip() == '' or (l.lstrip())[0] != '*'))
- starless = all(l.startswith(last_indent) or l.strip() == '' for l in lines[1:])
- # first line always indented
- self.print_token(current_token, lines[0])
- for line in lines[1:]:
- self.print_newline(preserve_statement_flags = True)
- if javadoc:
- # javadoc: reformat and re-indent
- self.print_token(current_token, ' ' + line.lstrip())
- elif starless and len(line) > last_indent_length:
- # starless: re-indent non-empty content, avoiding trim
- self.print_token(current_token, line[last_indent_length:])
- else:
- # normal comments output raw
- self.output.add_token(line)
- self.print_newline(preserve_statement_flags = preserve_statement_flags)
- def handle_comment(self, current_token, preserve_statement_flags):
- if current_token.wanted_newline:
- self.print_newline(preserve_statement_flags = preserve_statement_flags)
- if not current_token.wanted_newline:
- self.output.trim(True)
- self.output.space_before_token = True
- self.print_token(current_token)
- self.print_newline(preserve_statement_flags = preserve_statement_flags)
- def handle_dot(self, current_token):
- if self.start_of_statement(current_token):
- # The conditional starts the statement if appropriate.
- pass
- else:
- self.handle_whitespace_and_comments(current_token, True)
- if self.last_type == 'TK_RESERVED' and self.is_special_word(self.flags.last_text):
- self.output.space_before_token = True
- else:
- # allow preserved newlines before dots in general
- # force newlines on dots after close paren when break_chained - for bar().baz()
- self.allow_wrap_or_preserved_newline(current_token,
- self.flags.last_text == ')' and self.opts.break_chained_methods)
- self.print_token(current_token)
- def handle_unknown(self, current_token, preserve_statement_flags):
- self.print_token(current_token)
- if current_token.text[-1] == '\n':
- self.print_newline(preserve_statement_flags = preserve_statement_flags)
- def handle_eof(self, current_token):
- # Unwind any open statements
- while self.flags.mode == MODE.Statement:
- self.restore_mode()
- self.handle_whitespace_and_comments(current_token)
- def mkdir_p(path):
- try:
- if path:
- os.makedirs(path)
- except OSError as exc: # Python >2.5
- if exc.errno == errno.EEXIST and os.path.isdir(path):
- pass
- else:
- raise Exception()
- # Using object instead of string to allow for later expansion of info about each line
- class OutputLine:
- def __init__(self, parent):
- self.__parent = parent
- self.__character_count = 0
- self.__indent_count = -1
- self.__items = []
- self.__empty = True
- def get_character_count(self):
- return self.__character_count
- def is_empty(self):
- return self.__empty
- def set_indent(self, level):
- self.__character_count = self.__parent.baseIndentLength + level * self.__parent.indent_length
- self.__indent_count = level;
- def last(self):
- if not self.is_empty():
- return self.__items[-1]
- else:
- return None
- def push(self, input):
- self.__items.append(input)
- self.__character_count += len(input)
- self.__empty = False
- def pop(self):
- item = None
- if not self.is_empty():
- item = self.__items.pop()
- self.__character_count -= len(item)
- self.__empty = len(self.__items) == 0
- return item
- def remove_indent(self):
- if self.__indent_count > 0:
- self.__indent_count -= 1
- self.__character_count -= self.__parent.indent_length
- def trim(self):
- while self.last() == ' ':
- item = self._items.pop()
- self.__character_count -= 1
- self.__empty = len(self.__items) == 0
- def toString(self):
- result = ''
- if not self.is_empty():
- if self.__indent_count >= 0:
- result = self.__parent.indent_cache[self.__indent_count]
- result += ''.join(self.__items)
- return result
- class Output:
- def __init__(self, indent_string, baseIndentString = ''):
- self.indent_string = indent_string
- self.baseIndentString = baseIndentString
- self.indent_cache = [ baseIndentString ]
- self.baseIndentLength = len(baseIndentString)
- self.indent_length = len(indent_string)
- self.raw = False
- self.lines = []
- self.previous_line = None
- self.current_line = None
- self.space_before_token = False
- self.add_outputline()
- def add_outputline(self):
- self.previous_line = self.current_line
- self.current_line = OutputLine(self)
- self.lines.append(self.current_line)
- def get_line_number(self):
- return len(self.lines)
- def add_new_line(self, force_newline):
- if len(self.lines) == 1 and self.just_added_newline():
- # no newline on start of file
- return False
- if force_newline or not self.just_added_newline():
- if not self.raw:
- self.add_outputline()
- return True
- return False
- def get_code(self):
- sweet_code = "\n".join(line.toString() for line in self.lines)
- return re.sub('[\r\n\t ]+$', '', sweet_code)
- def set_indent(self, level):
- # Never indent your first output indent at the start of the file
- if len(self.lines) > 1:
- while level >= len(self.indent_cache):
- self.indent_cache.append(self.indent_cache[-1] + self.indent_string)
- self.current_line.set_indent(level)
- return True
- self.current_line.set_indent(0)
- return False
- def add_raw_token(self, token):
- for _ in range(token.newlines):
- self.add_outputline()
- self.current_line.push(token.whitespace_before)
- self.current_line.push(token.text)
- self.space_before_token = False
- def add_token(self, printable_token):
- self.add_space_before_token()
- self.current_line.push(printable_token)
- def add_space_before_token(self):
- if self.space_before_token and not self.just_added_newline():
- self.current_line.push(' ')
- self.space_before_token = False
- def remove_redundant_indentation(self, frame):
- # This implementation is effective but has some issues:
- # - can cause line wrap to happen too soon due to indent removal
- # after wrap points are calculated
- # These issues are minor compared to ugly indentation.
- if frame.multiline_frame or frame.mode == MODE.ForInitializer or frame.mode == MODE.Conditional:
- return
- # remove one indent from each line inside this section
- index = frame.start_line_index
- while index < len(self.lines):
- self.lines[index].remove_indent()
- index += 1
- def trim(self, eat_newlines = False):
- self.current_line.trim()
- while eat_newlines and len(self.lines) > 1 and self.current_line.is_empty():
- self.lines.pop()
- self.current_line = self.lines[-1]
- self.current_line.trim()
- if len(self.lines) > 1:
- self.previous_line = self.lines[-2]
- else:
- self.previous_line = None
- def just_added_newline(self):
- return self.current_line.is_empty()
- def just_added_blankline(self):
- if self.just_added_newline():
- if len(self.lines) == 1:
- return True
- line = self.lines[-2]
- return line.is_empty()
- return False
- class InputScanner:
- def __init__(self, input):
- self.__input = input
- self.__input_length = len(self.__input)
- self.__position = 0
- def back(self):
- self.__position -= 1
- def hasNext(self):
- return self.__position < self.__input_length
- def next(self):
- val = None
- if self.hasNext():
- val = self.__input[self.__position]
- self.__position += 1
- return val;
- def peek(self, index = 0):
- val = None
- index += self.__position;
- if index >= 0 and index < self.__input_length:
- val = self.__input[index];
- return val;
- def peekCharCode(self, index = 0):
- val = 0
- index += self.__position;
- if index >= 0 and index < self.__input_length:
- val = ord(self.__input[index])
- return val
- def test(self, pattern, index = 0):
- index += self.__position;
- return index >= 0 and index < self.__input_length and pattern.match(self.__input, index)
- def testChar(self, pattern, index = 0):
- val = self.peek(index)
- return val != None and pattern.match(val)
- def match(self, pattern):
- pattern_match = None
- if self.hasNext():
- pattern_match = pattern.match(self.__input, self.__position)
- if pattern_match:
- self.__position += len(pattern_match.group(0));
- return pattern_match
- class Tokenizer:
- whitespace = ["\n", "\r", "\t", " "]
- digit = re.compile('[0-9]')
- digit_bin = re.compile('[01]')
- digit_oct = re.compile('[01234567]')
- digit_hex = re.compile('[0123456789abcdefABCDEF]')
- positionable_operators = '!= !== % & && * ** + - / : < << <= == === > >= >> >>> ? ^ | ||'.split(' ')
- punct = (positionable_operators +
- # non-positionable operators - these do not follow operator position settings
- '! %= &= *= **= ++ += , -- -= /= :: <<= = => >>= >>>= ^= |= ~ ...'.split(' '))
- # Words which always should start on a new line
- line_starters = 'continue,try,throw,return,var,let,const,if,switch,case,default,for,while,break,function,import,export'.split(',')
- reserved_words = line_starters + ['do', 'in', 'of', 'else', 'get', 'set', 'new', 'catch', 'finally', 'typeof', 'yield', 'async', 'await', 'from', 'as']
- def __init__ (self, input_string, opts, indent_string):
- self.input = InputScanner(input_string)
- self.opts = opts
- self.indent_string = indent_string
- self.acorn = Acorn()
- # /* ... */ comment ends with nearest */ or end of file
- self.block_comment_pattern = re.compile('([\s\S]*?)((?:\*\/)|$)')
- # comment ends just before nearest linefeed or end of file
- self.comment_pattern = re.compile(self.acorn.six.u('([^\n\r\u2028\u2029]*)'))
- self.directives_block_pattern = re.compile('\/\* beautify( \w+[:]\w+)+ \*\/')
- self.directive_pattern = re.compile(' (\w+)[:](\w+)')
- self.directives_end_ignore_pattern = re.compile('([\s\S]*?)((?:\/\*\sbeautify\signore:end\s\*\/)|$)')
- self.template_pattern = re.compile('((<\?php|<\?=)[\s\S]*?\?>)|(<%[\s\S]*?%>)')
- def tokenize(self):
- self.in_html_comment = False
- self.tokens = []
- next = None
- last = None
- open = None
- open_stack = []
- comments = []
- while not (not last == None and last.type == 'TK_EOF'):
- token_values = self.__tokenize_next()
- next = Token(token_values[1], token_values[0], self.n_newlines, self.whitespace_before_token)
- while next.type == 'TK_COMMENT' or next.type == 'TK_BLOCK_COMMENT' or next.type == 'TK_UNKNOWN':
- if next.type == 'TK_BLOCK_COMMENT':
- next.directives = token_values[2]
- comments.append(next)
- token_values = self.__tokenize_next()
- next = Token(token_values[1], token_values[0], self.n_newlines, self.whitespace_before_token)
- if len(comments) > 0:
- next.comments_before = comments
- comments = []
- if next.type == 'TK_START_BLOCK' or next.type == 'TK_START_EXPR':
- next.parent = last
- open_stack.append(open)
- open = next
- elif (next.type == 'TK_END_BLOCK' or next.type == 'TK_END_EXPR') and \
- (not open == None and ( \
- (next.text == ']' and open.text == '[') or \
- (next.text == ')' and open.text == '(') or \
- (next.text == '}' and open.text == '{'))):
- next.parent = open.parent
- next.opened = open
- open = open_stack.pop()
- self.tokens.append(next)
- last = next
- return self.tokens
- def get_directives (self, text):
- if not self.directives_block_pattern.match(text):
- return None
- directives = {}
- directive_match = self.directive_pattern.search(text)
- while directive_match:
- directives[directive_match.group(1)] = directive_match.group(2)
- directive_match = self.directive_pattern.search(text, directive_match.end())
- return directives
- def __tokenize_next(self):
- whitespace_on_this_line = []
- self.n_newlines = 0
- self.whitespace_before_token = ''
- c = self.input.next()
- if c == None:
- return '', 'TK_EOF'
- if len(self.tokens) > 0:
- last_token = self.tokens[-1]
- else:
- # For the sake of tokenizing we can pretend that there was on open brace to start
- last_token = Token('TK_START_BLOCK', '{')
- while c in self.whitespace:
- if self.acorn.newline.match(c):
- # treat \r\n as one newline
- if not (c == '\n' and self.input.peek(-2) == '\r'):
- self.n_newlines += 1
- whitespace_on_this_line = []
- else:
- whitespace_on_this_line.append(c)
- c = self.input.next()
- if c == None:
- return '', 'TK_EOF'
- if len(whitespace_on_this_line) != 0:
- self.whitespace_before_token = ''.join(whitespace_on_this_line)
- if self.digit.match(c) or (c == '.' and self.input.testChar(self.digit)):
- allow_decimal = True
- allow_e = True
- local_digit = self.digit
- if c == '0' and self.input.testChar(re.compile('[XxOoBb]')):
- # switch to hex/oct/bin number, no decimal or e, just hex/oct/bin digits
- allow_decimal = False
- allow_e = False
- if self.input.testChar(re.compile('[Bb]')):
- local_digit = self.digit_bin
- elif self.input.testChar(re.compile('[Oo]')):
- local_digit = self.digit_oct
- else:
- local_digit = self.digit_hex
- c += self.input.next()
- elif c == '.':
- # Already have a decimal for this literal, don't allow another
- allow_decimal = False
- else:
- # we know this first loop will run. It keeps the logic simpler.
- c = ''
- self.input.back()
- # Add the digits
- while self.input.testChar(local_digit):
- c += self.input.next()
- if allow_decimal and self.input.peek() == '.':
- c += self.input.next()
- allow_decimal = False
- # a = 1.e-7 is valid, so we test for . then e in one loop
- if allow_e and self.input.testChar(re.compile('[Ee]')):
- c += self.input.next()
- if self.input.testChar(re.compile('[+-]')):
- c += self.input.next()
- allow_e = False
- allow_decimal = False
- return c, 'TK_WORD'
- if self.acorn.isIdentifierStart(self.input.peekCharCode(-1)):
- if self.input.hasNext():
- while self.acorn.isIdentifierChar(self.input.peekCharCode()):
- c += self.input.next()
- if not self.input.hasNext():
- break
- if not (last_token.type == 'TK_DOT' \
- or (last_token.type == 'TK_RESERVED' and last_token.text in ['set', 'get'])) \
- and c in self.reserved_words:
- if c == 'in' or c == 'of': # in and of are operators, need to hack
- return c, 'TK_OPERATOR'
- return c, 'TK_RESERVED'
- return c, 'TK_WORD'
- if c in '([':
- return c, 'TK_START_EXPR'
- if c in ')]':
- return c, 'TK_END_EXPR'
- if c == '{':
- return c, 'TK_START_BLOCK'
- if c == '}':
- return c, 'TK_END_BLOCK'
- if c == ';':
- return c, 'TK_SEMICOLON'
- if c == '/':
- comment = ''
- inline_comment = True
- if self.input.peek() == '*': # peek /* .. */ comment
- self.input.next()
- comment_match = self.input.match(self.block_comment_pattern)
- comment = '/*' + comment_match.group(0)
- directives = self.get_directives(comment)
- if directives and directives.get('ignore') == 'start':
- comment_match = self.input.match(self.directives_end_ignore_pattern)
- comment += comment_match.group(0)
- comment = re.sub(self.acorn.allLineBreaks, '\n', comment)
- return comment, 'TK_BLOCK_COMMENT', directives
- if self.input.peek() == '/': # peek // comment
- self.input.next()
- comment_match = self.input.match(self.comment_pattern)
- comment = '//' + comment_match.group(0)
- return comment, 'TK_COMMENT'
- startXmlRegExp = re.compile('<()([-a-zA-Z:0-9_.]+|{[\s\S]+?}|!\[CDATA\[[\s\S]*?\]\])(\s+{[\s\S]+?}|\s+[-a-zA-Z:0-9_.]+|\s+[-a-zA-Z:0-9_.]+\s*=\s*(\'[^\']*\'|"[^"]*"|{[\s\S]+?}))*\s*(/?)\s*>')
- self.has_char_escapes = False
- if c == '`' or c == "'" or c == '"' or \
- ( \
- (c == '/') or \
- (self.opts.e4x and c == "<" and self.input.test(startXmlRegExp, -1)) \
- ) and ( \
- (last_token.type == 'TK_RESERVED' and last_token.text in ['return', 'case', 'throw', 'else', 'do', 'typeof', 'yield']) or \
- (last_token.type == 'TK_END_EXPR' and last_token.text == ')' and \
- last_token.parent and last_token.parent.type == 'TK_RESERVED' and last_token.parent.text in ['if', 'while', 'for']) or \
- (last_token.type in ['TK_COMMENT', 'TK_START_EXPR', 'TK_START_BLOCK', 'TK_END_BLOCK', 'TK_OPERATOR', \
- 'TK_EQUALS', 'TK_EOF', 'TK_SEMICOLON', 'TK_COMMA'])):
- sep = c
- esc = False
- esc1 = 0
- esc2 = 0
- resulting_string = c
- in_char_class = False
- if sep == '/':
- # handle regexp
- in_char_class = False
- while self.input.hasNext() and \
- (esc or in_char_class or self.input.peek()!= sep) and \
- not self.input.testChar(self.acorn.newline):
- resulting_string += self.input.peek()
- if not esc:
- esc = self.input.peek() == '\\'
- if self.input.peek() == '[':
- in_char_class = True
- elif self.input.peek() == ']':
- in_char_class = False
- else:
- esc = False
- self.input.next()
- elif self.opts.e4x and sep == '<':
- # handle e4x xml literals
- xmlRegExp = re.compile('[\s\S]*?<(\/?)([-a-zA-Z:0-9_.]+|{[\s\S]+?}|!\[CDATA\[[\s\S]*?\]\])(\s+{[\s\S]+?}|\s+[-a-zA-Z:0-9_.]+|\s+[-a-zA-Z:0-9_.]+\s*=\s*(\'[^\']*\'|"[^"]*"|{[\s\S]+?}))*\s*(/?)\s*>')
- self.input.back()
- xmlStr = ""
- match = self.input.match(xmlRegExp)
- if match:
- rootTag = match.group(2)
- rootTag = re.sub(r'^{\s+', '{', re.sub(r'\s+}$', '}', rootTag))
- isCurlyRoot = rootTag.startswith('{')
- depth = 0
- while (match):
- isEndTag = match.group(1)
- tagName = match.group(2)
- isSingletonTag = (match.groups()[-1] != "") or (match.group(2)[0:8] == "![CDATA[")
- if not isSingletonTag and (
- tagName == rootTag or (isCurlyRoot and re.sub(r'^{\s+', '{', re.sub(r'\s+}$', '}', tagName)))):
- if isEndTag:
- depth -= 1
- else:
- depth += 1
- xmlStr += match.group(0)
- if depth <= 0:
- break
- match = self.input.match(xmlRegExp)
- # if we didn't close correctly, keep unformatted.
- if not match:
- xmlStr += self.input.match(re.compile('[\s\S]*')).group(0)
- xmlStr = re.sub(self.acorn.allLineBreaks, '\n', xmlStr)
- return xmlStr, 'TK_STRING'
- else:
- # handle string
- def parse_string(self, resulting_string, delimiter, allow_unescaped_newlines = False, start_sub = None):
- esc = False
- while self.input.hasNext():
- current_char = self.input.peek()
- if not (esc or (current_char != delimiter and
- (allow_unescaped_newlines or not self.acorn.newline.match(current_char)))):
- break
- # Handle \r\n linebreaks after escapes or in template strings
- if (esc or allow_unescaped_newlines) and self.acorn.newline.match(current_char):
- if current_char == '\r' and self.input.peek(1) == '\n':
- self.input.next()
- current_char = self.input.peek()
- resulting_string += '\n'
- else:
- resulting_string += current_char
- if esc:
- if current_char == 'x' or current_char == 'u':
- self.has_char_escapes = True
- esc = False
- else:
- esc = current_char == '\\'
- self.input.next()
- if start_sub and resulting_string.endswith(start_sub):
- if delimiter == '`':
- resulting_string = parse_string(self, resulting_string, '}', allow_unescaped_newlines, '`')
- else:
- resulting_string = parse_string(self, resulting_string, '`', allow_unescaped_newlines, '${')
- if self.input.hasNext():
- resulting_string += self.input.next()
- return resulting_string
- if sep == '`':
- resulting_string = parse_string(self, resulting_string, '`', True, '${')
- else:
- resulting_string = parse_string(self, resulting_string, sep)
- if self.has_char_escapes and self.opts.unescape_strings:
- resulting_string = self.unescape_string(resulting_string)
- if self.input.peek() == sep:
- resulting_string += self.input.next()
- if sep == '/':
- # regexps may have modifiers /regexp/MOD, so fetch those too
- # Only [gim] are valid, but if the user puts in garbage, do what we can to take it.
- while self.input.hasNext() and self.acorn.isIdentifierStart(self.input.peekCharCode()):
- resulting_string += self.input.next()
- resulting_string = re.sub(self.acorn.allLineBreaks, '\n', resulting_string)
- return resulting_string, 'TK_STRING'
- if c == '#':
- # she-bang
- if len(self.tokens) == 0 and self.input.peek() == '!':
- resulting_string = c
- while self.input.hasNext() and c != '\n':
- c = self.input.next()
- resulting_string += c
- return resulting_string.strip() + '\n', 'TK_UNKNOWN'
- # Spidermonkey-specific sharp variables for circular references
- # https://developer.mozilla.org/En/Sharp_variables_in_JavaScript
- # http://mxr.mozilla.org/mozilla-central/source/js/src/jsscan.cpp around line 1935
- sharp = '#'
- if self.input.hasNext() and self.input.testChar(self.digit):
- while True:
- c = self.input.next()
- sharp += c
- if (not self.input.hasNext()) or c == '#' or c == '=':
- break
- if c == '#':
- pass
- elif self.input.peek() == '[' and self.input.peek(1) == ']':
- sharp += '[]'
- self.input.next()
- self.input.next()
- elif self.input.peek() == '{' and self.input.peek(1) == '}':
- sharp += '{}'
- self.input.next()
- self.input.next()
- return sharp, 'TK_WORD'
- if c == '<' and self.input.peek() in ['?', '%']:
- self.input.back()
- template_match = self.input.match(self.template_pattern)
- if template_match:
- c = template_match.group(0)
- c = re.sub(self.acorn.allLineBreaks, '\n', c)
- return c, 'TK_STRING'
- if c == '<' and self.input.match(re.compile('\!--')):
- c = '<!--'
- while self.input.hasNext() and not self.input.testChar(self.acorn.newline):
- c += self.input.next()
- self.in_html_comment = True
- return c, 'TK_COMMENT'
- if c == '-' and self.in_html_comment and self.input.match(re.compile('->')):
- self.in_html_comment = False
- return '-->', 'TK_COMMENT'
- if c == '.':
- if self.input.peek() == '.' and self.input.peek(1) == '.':
- c += self.input.next() + self.input.next()
- return c, 'TK_OPERATOR'
- return c, 'TK_DOT'
- if c in self.punct:
- while self.input.hasNext() and c + self.input.peek() in self.punct:
- c += self.input.next()
- if not self.input.hasNext():
- break
- if c == ',':
- return c, 'TK_COMMA'
- if c == '=':
- return c, 'TK_EQUALS'
- return c, 'TK_OPERATOR'
- return c, 'TK_UNKNOWN'
- def unescape_string(self, s):
- # You think that a regex would work for this
- # return s.replace(/\\x([0-9a-f]{2})/gi, function(match, val) {
- # return String.fromCharCode(parseInt(val, 16));
- # })
- # However, dealing with '\xff', '\\xff', '\\\xff' makes this more fun.
- out = self.acorn.six.u('')
- escaped = 0
- input_scan = InputScanner(s)
- matched = None
- while input_scan.hasNext():
- # Keep any whitespace, non-slash characters
- # also keep slash pairs.
- matched = input_scan.match(re.compile(r'([\s]|[^\\]|\\\\)+'))
- if matched:
- out += matched.group(0)
- if input_scan.peek() != '\\':
- continue
- input_scan.next()
- if input_scan.peek() == 'x':
- matched = input_scan.match(re.compile('x([0-9A-Fa-f]{2})'))
- elif input_scan.peek() == 'u':
- matched = input_scan.match(re.compile('u([0-9A-Fa-f]{4})'));
- else:
- out += '\\'
- if input_scan.hasNext():
- out += input_scan.next()
- continue
- # If there's some error decoding, return the original string
- if not matched:
- return s
- escaped = int(matched.group(1), 16)
- if escaped > 0x7e and escaped <= 0xff and matched.group(0).startswith('x'):
- # we bail out on \x7f..\xff,
- # leaving whole string escaped,
- # as it's probably completely binary
- return s
- elif escaped >= 0x00 and escaped < 0x20:
- # leave 0x00...0x1f escaped
- out += '\\' + matched.group(0)
- continue
- elif escaped == 0x22 or escaped == 0x27 or escaped == 0x5c:
- # single-quote, apostrophe, backslash - escape these
- out += ('\\' + chr(escaped))
- else:
- out += self.acorn.six.unichr(escaped)
- return out
- def isFileDifferent(filepath, expected):
- try:
- return (''.join(io.open(filepath, 'rt', newline='').readlines()) != expected)
- except:
- return True
- def main():
- argv = sys.argv[1:]
- try:
- opts, args = getopt.getopt(argv, "s:c:e:o:rdEPjabkil:xhtfvXnCO:w:",
- ['indent-size=','indent-char=','eol=''outfile=', 'replace', 'disable-preserve-newlines',
- 'space-in-paren', 'space-in-empty-paren', 'jslint-happy', 'space-after-anon-function',
- 'brace-style=', 'keep-array-indentation', 'indent-level=', 'unescape-strings',
- 'help', 'usage', 'stdin', 'eval-code', 'indent-with-tabs', 'keep-function-indentation', 'version',
- 'e4x', 'end-with-newline','comma-first','operator-position=','wrap-line-length','editorconfig'])
- except getopt.GetoptError as ex:
- print(ex, file=sys.stderr)
- return usage(sys.stderr)
- js_options = default_options()
- file = None
- outfile = 'stdout'
- replace = False
- if len(args) == 1:
- file = args[0]
- for opt, arg in opts:
- if opt in ('--keep-array-indentation', '-k'):
- js_options.keep_array_indentation = True
- if opt in ('--keep-function-indentation','-f'):
- js_options.keep_function_indentation = True
- elif opt in ('--outfile', '-o'):
- outfile = arg
- elif opt in ('--replace', '-r'):
- replace = True
- elif opt in ('--indent-size', '-s'):
- js_options.indent_size = int(arg)
- elif opt in ('--indent-char', '-c'):
- js_options.indent_char = arg
- elif opt in ('--eol', '-e'):
- js_options.eol = arg
- elif opt in ('--indent-with-tabs', '-t'):
- js_options.indent_with_tabs = True
- elif opt in ('--disable-preserve-newlines', '-d'):
- js_options.preserve_newlines = False
- elif opt in ('--space-in-paren', '-P'):
- js_options.space_in_paren = True
- elif opt in ('--space-in-empty-paren', '-E'):
- js_options.space_in_empty_paren = True
- elif opt in ('--jslint-happy', '-j'):
- js_options.jslint_happy = True
- elif opt in ('--space_after_anon_function', '-a'):
- js_options.space_after_anon_function = True
- elif opt in ('--eval-code'):
- js_options.eval_code = True
- elif opt in ('--brace-style', '-b'):
- js_options.brace_style = arg
- elif opt in ('--unescape-strings', '-x'):
- js_options.unescape_strings = True
- elif opt in ('--e4x', '-X'):
- js_options.e4x = True
- elif opt in ('--end-with-newline', '-n'):
- js_options.end_with_newline = True
- elif opt in ('--comma-first', '-C'):
- js_options.comma_first = True
- elif opt in ('--operator-position', '-O'):
- js_options.operator_position = sanitizeOperatorPosition(arg)
- elif opt in ('--wrap-line-length ', '-w'):
- js_options.wrap_line_length = int(arg)
- elif opt in ('--stdin', '-i'):
- file = '-'
- elif opt in ('--editorconfig'):
- js_options.editorconfig = True
- elif opt in ('--version', '-v'):
- return print(__version__)
- elif opt in ('--help', '--usage', '-h'):
- return usage()
- if not file:
- file = '-'
- try:
- if outfile == 'stdout' and replace and not file == '-':
- outfile = file
- # Editorconfig used only on files, not stdin
- if getattr(js_options, 'editorconfig'):
- editorconfig_filepath = file
- if editorconfig_filepath == '-':
- if outfile != 'stdout':
- editorconfig_filepath = outfile
- else:
- fileType = 'js'
- editorconfig_filepath = 'stdin.' + fileType
- # debug("EditorConfig is enabled for ", editorconfig_filepath);
- js_options = copy.copy(js_options)
- set_file_editorconfig_opts(editorconfig_filepath, js_options)
- pretty = beautify_file(file, js_options)
- if outfile == 'stdout':
- # python automatically converts newlines in text to "\r\n" when on windows
- # switch to binary to prevent this
- if sys.platform == "win32":
- import msvcrt
- msvcrt.setmode(sys.stdout.fileno(), os.O_BINARY)
- sys.stdout.write(pretty)
- else:
- if isFileDifferent(outfile, pretty):
- mkdir_p(os.path.dirname(outfile))
- # python automatically converts newlines in text to "\r\n" when on windows
- # set newline to empty to prevent this
- with io.open(outfile, 'wt', newline='') as f:
- print('writing ' + outfile, file=sys.stderr)
- try:
- f.write(pretty)
- except TypeError:
- # This is not pretty, but given how we did the version import
- # it is the only way to do this without having setup.py fail on a missing six dependency.
- six = __import__("six")
- f.write(six.u(pretty))
- except Exception as ex:
- print(ex, file=sys.stderr)
- return 1
- # Success
- return 0
|