.. _modules: Modules ####### Modules provide a way of organizing QL code by grouping together related types, predicates and other modules. You can import modules into other files, which avoids duplication, and helps structure your code into more manageable pieces. .. _defining-module: Defining a module ***************** There are various ways to define modules—here is an example of the simplest way, declaring an :ref:`explicit module ` named ``Example`` containing a class ``OneTwoThree``:: module Example { class OneTwoThree extends int { OneTwoThree() { this = 1 or this = 2 or this = 3 } } } The name of a module can be any `identifier `_ that starts with an uppercase or lowercase letter. ``.ql`` or ``.qll`` files also implicitly define modules. Read more about the different :ref:`kinds of modules ` below. You can also annotate a module. See the list of :ref:`annotations ` available for modules. Note that you can only annotate :ref:`explicit modules `. File modules cannot be annotated. .. _kinds-of-modules: Kinds of modules **************** File modules ============ Each query file (extension ``.ql``) and library file (extension ``.qll``) implicitly defines a module. The module has the same name as the file, but any spaces in the file name are replaced by underscores (``_``). The contents of the file form the :ref:`body of the module `. .. _library-modules: Library modules --------------- A library module is defined by a ``.qll`` file. It can contain any of the elements listed in :ref:`module-bodies` below, apart from select clauses. For example, consider the following QL library: **OneTwoThreeLib.qll** :: class OneTwoThree extends int { OneTwoThree() { this = 1 or this = 2 or this = 3 } } This file defines a library module named ``OneTwoThreeLib``. The body of this module defines the class ``OneTwoThree``. .. _query-modules: Query modules ------------- A query module is defined by a ``.ql`` file. It can contain any of the elements listed in :ref:`module-bodies` below. The difference is that a query module must have at least one query in its :ref:`namespace `. This is usually a :ref:`select clause `, but can also be a :ref:`query predicate `. For example: **OneTwoQuery.ql** :: import OneTwoThreeLib from OneTwoThree ott where ott = 1 or ott = 2 select ott This file defines a query module named ``OneTwoQuery``. The body of this module consists of an :ref:`import statement ` and a :ref:`select clause `. .. _explicit-modules: Explicit modules ================ You can also define a module within another module. This is an explicit module definition. An explicit module is defined with the keyword ``module`` followed by the module name, and then the module body enclosed in braces. It can contain any of the elements listed in :ref:`module-bodies` below, apart from select clauses. For example, you could add the following QL snippet to the library file **OneTwoThreeLib.qll** defined :ref:`above `:: ... module M { class OneTwo extends OneTwoThree { OneTwo() { this = 1 or this = 2 } } } This defines an explicit module named ``M``. The body of this module defines the class ``OneTwo``. .. _module-bodies: Module bodies ************* The body of a module is the code inside the module definition, for example the class ``OneTwo`` in the :ref:`explicit module ` ``M``. In general, the body of a module can contain the following constructs: - :ref:`import-statements` - :ref:`predicates` - :ref:`types` (including user-defined :ref:`classes `) - :ref:`aliases` - :ref:`explicit-modules` - :ref:`select-clauses` (only available in a :ref:`query module `) .. index:: import .. _importing-modules: Importing modules ***************** The main benefit of storing code in a module is that you can reuse it in other modules. To access the contents of an external module, you can import the module using an :ref:`import statement `. When you import a module this brings all the names in its namespace, apart from :ref:`private` names, into the :ref:`namespace ` of the current module. .. _import-statements: Import statements ================= Import statements are used for importing modules and are of the form:: import as import Import statements are usually listed at the beginning of the module. Each import statement imports one module. You can import multiple modules by including multiple import statements (one for each module you want to import). An import statement can also be :ref:`annotated ` with ``private``. You can import a module under a different name using the ``as`` keyword, for example ``import javascript as js``. The ```` itself can be a module name, a selection, or a qualified reference. See :ref:`name-resolution` for more details.