Added lexars
This commit is contained in:
parent
382b2ce94b
commit
f9f64b0558
820 changed files with 149371 additions and 0 deletions
lexers/testdata
75
lexers/testdata/fennel.actual
vendored
Normal file
75
lexers/testdata/fennel.actual
vendored
Normal file
|
@ -0,0 +1,75 @@
|
|||
;; An example of some possible linters using Fennel's --plugin option.
|
||||
|
||||
;; The first two linters here can only function on static module
|
||||
;; use. For instance, this code can be checked because they use static
|
||||
;; field access on a local directly bound to a require call:
|
||||
|
||||
;; (local m (require :mymodule))
|
||||
;; (print m.field) ; fails if mymodule lacks a :field field
|
||||
;; (print (m.function 1 2 3)) ; fails unless mymodule.function takes 3 args
|
||||
|
||||
;; However, these cannot:
|
||||
|
||||
;; (local m (do (require :mymodule)) ; m is not directly bound
|
||||
;; (print (. m field)) ; not a static field reference
|
||||
;; (let [f m.function]
|
||||
;; (print (f 1 2 3)) ; intermediate local, not a static field call on m
|
||||
|
||||
;; Still, pretty neat, huh?
|
||||
|
||||
;; This file is provided as an example and is not part of Fennel's public API.
|
||||
|
||||
(fn save-require-meta [from to scope]
|
||||
"When destructuring, save module name if local is bound to a `require' call.
|
||||
Doesn't do any linting on its own; just saves the data for other linters."
|
||||
(when (and (sym? to) (not (multi-sym? to)) (list? from)
|
||||
(sym? (. from 1)) (= :require (tostring (. from 1)))
|
||||
(= :string (type (. from 2))))
|
||||
(let [meta (. scope.symmeta (tostring to))]
|
||||
(set meta.required (tostring (. from 2))))))
|
||||
|
||||
(fn check-module-fields [symbol scope]
|
||||
"When referring to a field in a local that's a module, make sure it exists."
|
||||
(let [[module-local field] (or (multi-sym? symbol) [])
|
||||
module-name (-?> scope.symmeta (. (tostring module-local)) (. :required))
|
||||
module (and module-name (require module-name))]
|
||||
(assert-compile (or (= module nil) (not= (. module field) nil))
|
||||
(string.format "Missing field %s in module %s"
|
||||
(or field :?) (or module-name :?)) symbol)))
|
||||
|
||||
(fn arity-check? [module] (-?> module getmetatable (. :arity-check?)))
|
||||
|
||||
(fn arity-check-call [[f & args] scope]
|
||||
"Perform static arity checks on static function calls in a module."
|
||||
(let [arity (# args)
|
||||
last-arg (. args arity)
|
||||
[f-local field] (or (multi-sym? f) [])
|
||||
module-name (-?> scope.symmeta (. (tostring f-local)) (. :required))
|
||||
module (and module-name (require module-name))]
|
||||
(when (and (arity-check? module) _G.debug _G.debug.getinfo
|
||||
(not (varg? last-arg)) (not (list? last-arg)))
|
||||
(assert-compile (= (type (. module field)) :function)
|
||||
(string.format "Missing function %s in module %s"
|
||||
(or field :?) module-name) f)
|
||||
(match (_G.debug.getinfo (. module field))
|
||||
{: nparams :what "Lua" :isvararg true}
|
||||
(assert-compile (<= nparams (# args))
|
||||
(: "Called %s.%s with %s arguments, expected %s+"
|
||||
:format f-local field arity nparams) f)
|
||||
{: nparams :what "Lua" :isvararg false}
|
||||
(assert-compile (= nparams (# args))
|
||||
(: "Called %s.%s with %s arguments, expected %s"
|
||||
:format f-local field arity nparams) f)))))
|
||||
|
||||
(fn check-unused [ast scope]
|
||||
(each [symname (pairs scope.symmeta)]
|
||||
(assert-compile (or (. scope.symmeta symname :used) (symname:find "^_"))
|
||||
(string.format "unused local %s" (or symname :?)) ast)))
|
||||
|
||||
{:destructure save-require-meta
|
||||
:symbol-to-expression check-module-fields
|
||||
:call arity-check-call
|
||||
;; Note that this will only check unused args inside functions and let blocks,
|
||||
;; not top-level locals of a chunk.
|
||||
:fn check-unused
|
||||
:do check-unused}
|
Loading…
Add table
Add a link
Reference in a new issue