Relational databases consist of one or more tables that each have a plurality of columns or fields. Queries used to retrieve information from such tables are relatively straightforward to write and involve selecting one or more fields to be returned based on values in those or other fields in the tables. Relational databases have been available for a long time; so many application developers know how to write queries for relational databases.
Multidimensional databases (a more recent and fast growing development in database technologies) have been identified and accepted as being more advantageous than relational databases in data aggregation, online analytical data processing (OLAP) and end user reporting scenarios, all of which are common functional requirements in business information systems today. In multidimensional databases, data from relational databases is aggregated into a data structure known as a cube that facilitates more efficient multidimensional analysis of key business metric values across a set of defined business dimensions. A cube includes a set of Dimensions and Measures whose data values are typically sourced from relational databases. Dimensions are categories by which key business metrics can be aggregated, grouped, and analyzed (Example: A Sales amount business metric can be grouped and analyzed by dimensions like Customer, Salesperson, Sales region etc) . A Dimension has attributes (Example: A Customer dimension can contain attributes like Customer Name, State, City, Gender etc) that can be optionally organized into hierarchies that have different levels (Example: A Customer Dimension can contain a State-City-Customer hierarchy comprising the levels State, City, and Customer Name). Measures, which represent aggregated business metrics stored in the cube, can be grouped and analyzed by the cube dimensions.
Specialized query languages exist to facilitate the querying of multidimensional databases. The most common and widely accepted/implemented query language for querying multidimensional databases is MDX. MDX is primarily a specification developed and introduced by Microsoft in 1998. Microsoft proposed that MDX be a standard, and its adoption among application writers and other OLAP providers is steadily increasing. Products like Microsoft SQL Analysis Server, IBM's DB2 etc. implement MDX as their OLAP query language. There are also some multidimensional database vendors who implement and support proprietary query language syntax for querying data stored in their cube artifacts. Query languages for querying multidimensional data stored in cubes provide a rich set of functions and operators that can be used to define expressions for common analytical computations like ratios, trends, averages, time series calculations etc. Expertise in using these query languages is not a common skill and requires a fair amount of investment in training to develop (even for developers who are proficient in relational databases and the SQL query language)
The MDX language for instance has a very exacting syntax and the majority of application developers have little to no experience in using it. Writing MDX queries is time consuming, prone to errors, and a skill that takes a significant amount of time to acquire. In particular, each of the large number of MDX functions requires an exacting syntax and a strong knowledge of the underlying cube structure. For example, the function that returns the top n percent requires that a path to a set, a percent value, and a measure be provided in a particular order and with a particular syntax. Another common requirement like computing a Year to date running total of a business metric will require the use of multiple MDX functions and operators to define the related expression. The cube data members referenced by the functions also need to be specified using an exacting syntax.
A problem trend observed here is that while the analytics computational requirements (Examples: Ratios, Moving Averages, Year to date totals etc) are well known to most business application developers and users, specialized skills are required to implement their definitions in code.
Further, when used in application code, analytics expressions appear as strings in code written by application developers. These strings cannot be evaluated at compile time. As a result, if the analytics expression is incorrectly written by the application developer, an error will not appear until run time. This complicates the development of the application code and makes it more likely that a code “bug” will ship in the application. The other side effect of these expressions being coded as strings is that they will not be resilient to changes introduced to the structure of the cubes that they are based on. Breaking impact of a change to the structure of a cube referenced by an expression cannot be identified at compile time and will commonly result in runtime errors. Developers are required to execute string-based find and patch steps to fix the expressions in code, which is not an optimal process and is one that is error prone.
Though MDX was cited as an example here, these problems/challenges also apply to other proprietary multidimensional query languages.
Currently, some systems provide templates of commonly used Analytics expressions. However these templates are again simply strings. Since these templates are written without any knowledge of the underlying cube structure, the application programmer must still know all of the names of the different structural elements of the cube and enter references to them correctly into the code. Further, if the user wants to apply a template to different scenarios, he would need to create multiple instances of the template with hard coded parameter bindings for each of the measures and dimension levels. There is no easy way to define a template once and vary its parameter bindings dynamically based on need. Additionally, compile time verification and resilience to changes in the underlying cube structures continue to be issues when using these templates.
The discussion above is merely provided for general background information and is not intended to be used as an aid in determining the scope of the claimed subject matter.
Abstracted metadata is generated for multidimensional analytics expression templates. This Metadata is used to define analytics queries for an application using a design time query builder (the resulting query definitions are stored as application metadata) and to code generate strongly typed expression classes that are compiled into an expression assembly. The expression classes can be referenced and used to also define analytics queries in code. The use of strongly-typed expression classes in code enables the benefits of compile time verifications and the elimination of the usage of strings to define analytics queries in application code. The Expression assembly is used at runtime to resolve the expressions used in analytics queries. When resolving an expression, an abstract expression tree is returned by the expression assembly. This expression tree is translated to a native query language expression (Example: MDX expression) by an expression translator. Multiple expression translators can be implemented to target different multidimensional databases and query languages.
This Summary is provided to introduce a selection of concepts in a simplified form that are further described below in the Detailed Description. This Summary is not intended to identify key features or essential features of the claimed subject matter, nor is it intended to be used as an aid in determining the scope of the claimed subject matter.
Embodiments are operational with numerous other general purpose or special purpose computing system environments or configurations. Examples of well-known computing systems, environments, and/or configurations that may be suitable for use with various embodiments include, but are not limited to, personal computers, server computers, hand-held or laptop devices, multiprocessor systems, microprocessor-based systems, set top boxes, programmable consumer electronics, network PCs, minicomputers, mainframe computers, telephony systems, distributed computing environments that include any of the above systems or devices, and the like.
Embodiments may be described in the general context of computer-executable instructions, such as program modules, being executed by a computer. Generally, program modules include routines, programs, objects, components, data structures, etc. that perform particular tasks or implement particular abstract data types. Some embodiments are designed to be practiced in distributed computing environments where tasks are performed by remote processing devices that are linked through a communications network. In a distributed computing environment, program modules are located in both local and remote computer storage media including memory storage devices.
With reference to
Computer 110 typically includes a variety of computer readable media. Computer readable media can be any available media that can be accessed by computer 110 and includes both volatile and nonvolatile media, removable and non-removable media. By way of example, and not limitation, computer readable media may comprise computer storage media and communication media. Computer storage media includes both volatile and nonvolatile, removable and non-removable media implemented in any method or technology for storage of information such as computer readable instructions, data structures, program modules or other data. Computer storage media includes, but is not limited to, RAM, ROM, EEPROM, flash memory or other memory technology, CD-ROM, digital versatile disks (DVD) or other optical disk storage, magnetic cassettes, magnetic tape, magnetic disk storage or other magnetic storage devices, or any other medium which can be used to store the desired information and which can be accessed by computer 110. Communication media typically embodies computer readable instructions, data structures, program modules or other data in a modulated data signal such as a carrier wave or other transport mechanism and includes any information delivery media. The term “modulated data signal” means a signal that has one or more of its characteristics set or changed in such a manner as to encode information in the signal. By way of example, and not limitation, communication media includes wired media such as a wired network or direct-wired connection, and wireless media such as acoustic, RF, infrared and other wireless media. Combinations of any of the above should also be included within the scope of computer readable media.
The system memory 130 includes computer storage media in the form of volatile and/or nonvolatile memory such as read only memory (ROM) 131 and random access memory (RAM) 132. A basic input/output system 133 (BIOS), containing the basic routines that help to transfer information between elements within computer 110, such as during start-up, is typically stored in ROM 131. RAM 132 typically contains data and/or program modules that are immediately accessible to and/or presently being operated on by processing unit 120. By way of example, and not limitation,
The computer 110 may also include other removable/non-removable volatile/nonvolatile computer storage media. By way of example only,
The drives and their associated computer storage media discussed above and illustrated in
A user may enter commands and information into the computer 110 through input devices such as a keyboard 162, a microphone 163, and a pointing device 161, such as a mouse, trackball or touch pad. Other input devices (not shown) may include a joystick, game pad, satellite dish, scanner, or the like. These and other input devices are often connected to the processing unit 120 through a user input interface 160 that is coupled to the system bus, but may be connected by other interface and bus structures, such as a parallel port, game port or a universal serial bus (USB). A monitor 191 or other type of display device is also connected to the system bus 121 via an interface, such as a video interface 190. In addition to the monitor, computers may also include other peripheral output devices such as speakers 197 and printer 196, which may be connected through an output peripheral interface 195.
The computer 110 is operated in a networked environment using logical connections to one or more remote computers, such as a remote computer 180. The remote computer 180 may be a personal computer, a hand-held device, a server, a router, a network PC, a peer device or other common network node, and typically includes many or all of the elements described above relative to the computer 110. The logical connections depicted in
When used in a LAN networking environment, the computer 110 is connected to the LAN 171 through a network interface or adapter 170. When used in a WAN networking environment, the computer 110 typically includes a modem 172 or other means for establishing communications over the WAN 173, such as the Internet. The modem 172, which may be internal or external, may be connected to the system bus 121 via the user input interface 160, or other appropriate mechanism. In a networked environment, program modules depicted relative to the computer 110, or portions thereof, may be stored in the remote memory storage device. By way of example, and not limitation,
Memory 204 is implemented as non-volatile electronic memory such as random access memory (RAM) with a battery back-up module (not shown) such that information stored in memory 204 is not lost when the general power to mobile device 200 is shut down. A portion of memory 204 is preferably allocated as addressable memory for program execution, while another portion of memory 204 is preferably used for storage, such as to simulate storage on a disk drive.
Memory 204 includes an operating system 212, application programs 214 as well as an object store 216. During operation, operating system 212 is preferably executed by processor 202 from memory 204. Operating system 212, in one preferred embodiment, is a WINDOWS® CE brand operating system commercially available from Microsoft Corporation. Operating system 212 is preferably designed for mobile devices, and implements database features that can be utilized by applications 214 through a set of exposed application programming interfaces and methods. The objects in object store 216 are maintained by applications 214 and operating system 212, at least partially in response to calls to the exposed application programming interfaces and methods.
Communication interface 208 represents numerous devices and technologies that allow mobile device 200 to send and receive information. The devices include wired and wireless modems, satellite receivers and broadcast tuners to name a few. Mobile device 200 can also be directly connected to a computer to exchange data therewith. In such cases, communication interface 208 can be an infrared transceiver or a serial or parallel communication connection, all of which are capable of transmitting streaming information.
Input/output components 206 include a variety of input devices such as a touch-sensitive screen, buttons, rollers, and a microphone as well as a variety of output devices including an audio generator, a vibrating device, and a display. The devices listed above are by way of example and need not all be present on mobile device 200. In addition, other input/output devices may be attached to or found with mobile device 200.
To make it easier for application developers to define multidimensional database analytics expressions at design time and in code, metadata descriptions of commonly used analytics expressions (in the form of expression templates) and programming object classes formed from the metadata descriptions are provided. Each expression object class generated from the metadata description will contain a method that can be invoked to return an abstract expression tree for an expression. The tree is passed to an expression translator for translation to the native query language of a target multidimensional database system. Each object class resolves expression parameter references to components of multidimensional database structures, such as measures, dimensions and levels within a cube, into full paths. Because the expression objects resolve components of multidimensional database structures into paths, application developers can define multidimensional database expressions by simply referencing the names of components in the structure's metadata (measures, dimensions, dimension attributes, hierarchies, levels) without knowing the full paths of the structural components expressed in a platform specific syntax. In addition, the reference to strongly typed structural components allows any errors in referencing the structural components to be detected at compile time rather than run time.
Although the methods and systems described herein can be practiced with any multidimensional database query language, for simplicity, the MDX query language is referenced below. In addition, although the present invention may be used with any multidimensional database structure, cube database structures are referenced below.
In step 300 of
At step 302, cube metadata 400 is provided to code generation application 402. Code generation 402 converts the metadata into cube classes 404, which in one embodiment are written in C#. Typically, cube classes 404 include a primary type for the cube itself and secondary types for each Dimension in the cube. In the primary class, the Measures for the cube are exposed as properties. Nested types are provided within the Dimension types to provide for Hierarchies, Levels and Members as defined in the cube metadata. The cube classes are derived from a cube base class (not shown).
At step 304, metadata is generated for an analytics expression template. In
where the Name attribute in the Template tag provides the name to be assigned to the class for the analytics expression and the Type attribute designates whether the analytics expression will return a single value or a set of values. In particular, “metric” is used as the type if the analytics expression will return a single aggregated numeric metric value (Example: Year To Date Sales) and “set” is used if the analytics expression will return a set of values (Example: Top 10 Customer names).
Each Parameter tag describes a parameter that forms part of the analytics expression. The Name attribute in the Parameter tags provides the name for the parameter and the Binding Type attribute in the Parameter tags describes which of a set of cube component data types the parameter belongs to. Possible cube component data types include Measure, Dimension, DimensionAttribute, Hierarchy, TimeHierarchy, Level, or Year Level. Other cube component data types are possible, with each cube component data type describing a component of the cube.
A string representation of the abstract analytics expression tree is shown between the Translation tags for readability. In an actual implementation, the expression tree would be stored as a BLOB (Binary Large Object) in the metadata and translated to a specific analytics query language like MDX by an expression translator. The numbered place holders {0}, {1} and {2} in the expression string represent parameter placeholders, which are filled with paths resolved from the parameters designated in the Parameter tags in the order in which the Parameter tags appear within the metadata. Thus, the Metric parameter would fill slot zero, the Calendar parameter would fill slot one, and the Year parameter would fill slot two in the example above.
The expression for the YTDTotal template above, utilizes an analytics function PeriodsToDate, which produces a set of Measure values with one value for each member of the YearLevel in the TimeHierarchy. The string also utilizes the function “Sum” to sum the values in the set.
Other examples of analytics expression metadata include:
Which provides the ratio of a measure at one level to the aggregation of the measures determined at the parent node of the level.
Which provides the difference for a measure between two years if there was a value for the measure the preceding year.
Which allocates a value based on the percentage of a measure that is allocated to a member relative to all members associated with a parent.
Which provides the top N members of a set ranked based on the specified metric.
Which provides the bottom N members of a set ranked based on the specified metric.
Which provides the Top N % of members of a set ranked based on the specified metric
Which provides the Bottom N % of members of a set ranked based on the specified metric
Which provides the average of the measures across a particular level or attribute of the cube.
Which provides a rolling average of a measure across a time hierarchy for a user specified window period
Which provides the percentage of a measure that a member contains relative to an entire dimension.
Which provides the percentage of a measure that a member contains relative to the sum of the measure across all children of the parent to the member.
Abstracted analytics functions are used to define expression templates. The abstract expression trees are translated by an expression translator to a specific analytics query language at runtime. The translated analytics query is understood and executed by the analytics query language interpreter of a multidimensional database to generate the result set. These functions are shown as examples only and any analytics functions supported by the analytics query language interpreter may be used in analytics expression metadata.
The analytics expression metadata is created using a user interface 410 such as user interface 500 of
An expandable parameter list 506 is provided with a parameter name field 508 and a type field 510. With each entry of a parameter name or type, a new row is provided to allow insertion of another parameter name and type. In text box 512, the user types in the analytics expression string with slot indicators for the parameters found in the parameter list. When save button 514 is pressed, the metadata is generated to produce an XML representation like the representations shown above. In creating the metadata from the user interface, the parameters are placed in the same order as they appear in the user interface.
After metadata 408 and 406 for the analytics expression have been generated, the metadata may be used with cube metadata 400 to generate analytics query metadata 412 that describes an analytics query at step 306. Under one specific embodiment, analytics query metadata 412 is formed using a drag-and-drop user interface that allows the developer to place cube components and expression templates in a query shell to define the query.
Each of the cube components and expressions shown in pane 600 may be selected, dragged, and dropped into a query shell in query pane 640. A query shell is a user interface element that consists of a query name, such as Sample Query 642, a Rows header, such as Rows header 644, a Columns header, such as Columns header 646, and a Data header such as data header 648. Cube components may be dragged/moved into the Rows and Columns headers. For example, the Calendar year level 612 has been dragged into Columns header 646. Measures and expression templates may be dragged/moved into the data header and into expression templates previously placed in the data header. For example, Year Ago Difference expression template 624 has been dragged into data header 648 in
In
When an element is moved from elements pane 600 to the query shell, the metadata definition for the element is inserted into the metadata definition for the query. Thus, when a cube component is moved onto the query shell, the metadata for that component is inserted into the query metadata. Similarly, when an analytics expression is moved onto the query shell, the metadata for the expression is inserted into the metadata for the query.
The queries defined through the analytics queries user interface 414 are stored as analytics query metadata 412. This metadata defines the name of the query, the dimensions, hierarchies, levels or sets defined for the rows and columns of the query, and the measures and expression templates defined for the data of the query.
Analytics query metadata 412 is stored along with user-defined analytics expression metadata 408, shipped analytics expression metadata 406 and cube metadata 400 in a runtime metadata store 416 which can be accessed during runtime to resolve queries.
In addition to using user-defined analytics expression metadata 408 to define analytics queries, the method of claim 3 uses the metadata to generate user-defined analytics expression classes 418 at step 308. Specifically, user-defined analytics expression metadata 408 is provide to code generation 420 to form user-defined analytics expression classes 418.
At step 704, private fields are defined for each parameter in the metadata. At step 706, a constructor method is written to the class. Under one embodiment, the constructor includes a parameter for each parameter defined in the metadata. In other embodiments, the constructor is overloaded by writing different versions of the constructor with different numbers of parameters. This allows the constructor to be called with different numbers of parameters. Within the constructor, setter functions are called for each parameter that is passed in.
At step 708, a property is formed for each parameter in the metadata. Each property includes setter and getter functions that respectively set the corresponding private field for the parameter equal to the value that the property is being set to and return the value of the property. The property is given the same name as the name attribute in the parameter of the metadata.
At step 710, a method to obtain an expression tree for the analytics expression template is added to the class. Under one embodiment, this method uses the name of the analytics expression template to locate the metadata for the analytics expression template. Using the metadata, the method constructs an expression tree, which is an abstract expression data structure containing the elements of the analytics expression template. This expression tree is an abstraction of the analytics expression in that it is not defined using any particular query language syntax. In order to produce a query string that represents the analytics expression, this abstract expression tree must be translated into a desired analytics query language string as described further below. Because the expression tree is an abstraction, many different translators may be written such that each translator would produce a different string with syntax for a different query language given the same expression tree.
Below is an example of an analytics expression class that is formed from the analytics metadata shown above for the YTDTotal function.
At step 310 of
In some embodiments of application development software 424, an auto-complete feature is provided that provides a list of available object extensions when an object class is typed into application source code 422 followed by a period. For example, if “Salescube.” was entered, the auto-complete feature would provide a list of subordinate types found within that class. Thus, for a cube object, the dimensions and hierarchies would be shown. This allows the programmer to select from a list of available sub-types and thus does not require the programmer to know or key in the components of the cube, or properties or methods of an analytics expression. Application development software 424 can also validate objects as they are written to application source code 422 to ensure that they correspond to a defined object in one of the classes.
As shown in
Since the analytics expressions are now strongly typed classes, and can accept references to the cube component data types, compiler 800 can determine if the references to the analytics expressions and the cube components are valid. If the references to the cube component data types or the analytics expressions are not valid, the compiler returns an error for the application source code. Compiler 422 can also check the calls used to set the values for the parameters in the expression, either the constructor calls or the parameter property calls, and can return a compile error if these calls are invalid. In particular, a compile error can be returned if the wrong cube component data type is being passed in a call based on the expected cube component data types defined in the expression class.
During a runtime 812, compiled code in application assembly 806 is executed. This code uses objects defined in cube assembly 802, user-defined analytics expressions assembly 804, shipped analytics expression assembly 808 and shipped analytics query framework assembly 810. Runtime 812 also makes use of runtime metadata store 814 and expression tree translator assembly 816.
Specifically, during runtime 812, when code in application assembly 806 encounters an analytics expression, it requests the expression tree for the expression from the expression's corresponding assembly in user-defined expression assembly 804 or shipped analytics expression assembly 808. This is done using the GetExpressionTree method of the expression's class. Runtime 812 then calls an expression tree translator in expression tree translator assembly 816 to obtain a string from the expression tree. Note that different translators can be provided for different query languages such that a single expression tree can be translated into different query languages depending on the translator that is used. After the query string has been formed, it is executed.
Similarly, database queries that are encountered during runtime 812 are converted into strings and then executed.
Before executing such a query, the metadata describing the query is first loaded. In some embodiments, this involves loading the metadata from metadata store 416. Once the metadata has been loaded, conversion of the query begins at step 900 where an element of the query is selected such as the Rows, Columns, or Data.
At step 902, the expression or variable found in the selected element is evaluated to determine if it is an analytics expression. If it is not an analytics expression, it is a cube component such as a Dimension, Hierarchy, Level or Measure. At step 904, the reference to the cube component is converted into a path string that identifies the cube component using the proper syntax for the query language. The method then continues at step 924 where the query metadata is examined to determine if there are more elements to process. If there are more elements, the process returns to step 900 to select the next element of the query.
If a selected element of the query is an analytics expression, an expression tree for the expression is requested at step 906. Typically, this is done using the GetExpressionTree method found in the class for the analytics expression.
At step 908, the expression tree is translated to obtain a string representing the analytics expression written in the proper syntax for the query language of interest. This is done by applying the expression to an expression tree translator, such as found in expression tree translator assembly 816.
As shown in
If a parameter is not an analytics expression at step 912, the parameter is a cube component. This cube component is converted into a path string that identifies the cube component using the proper syntax for the query language at step 916. This string is inserted into the appropriate slot of the current expression string. At step 914, the process determines if there are more parameters for the current expression tree. If there are more parameters, the next parameter is selected at step 910.
When there are no more parameters for the current expression tree, the process determines if it is at the top level of the recursion. If it is not at the top level of the recursion, expression trees above the current expression tree have not been completely processed. As such, the process returns up a level in the recursion at step 922 by setting the expression tree directly above the current expression tree as the current expression tree. The process then determines if there are more parameters in the new current expression tree by returning to step 918.
When the top level of the recursion is reached at step 920, the expression tree translation is complete and a full expression string has been formed from the analytics expression tree requested at step 906. The process then returns to step 924 to determine if there are more query elements to process. When there are no more query elements to process, the complete query string is executed at step 926.
Application source code 422 can include instructions to load and execute queries defined in analytics query metadata 412 or instructions that define a new analytics query and that bind parameter values to the query. Instructions that load and execute a query defined in metadata using the shipped analytics query framework classes. Specifically, the following instructions load and execute a query stored in metadata:
Instructions that define a new query and bind parameter values to the query use the cube classes and the analytics expression classes to define the elements of the query. The following example instructions define a new query and assign values to the parameters of the query.
These instructions associate the SalesTerritory dimension of the SalesCube cube with the Rows of the query and the OrderDate.Quarter level with the Columns of the query. The Data of the query is set using a call to the constructor of the YTDTotal analytics expression class. This constructor call includes parameter values for the two parameters of the YTDTotal analytics expression, where Salescube.Orderdate indicates the Time Hierarchy Orderdate, and Salescube.Salesamount indicates the measure in Salescube that is to be summed in the year-to-date expression.
The values for the Parameters can alternatively be set by calling properties of the expression class. For example, the instruction: “YTDTotal.Year=Salescube.Orderdate.Year” could be used to set the Year Parameter in the YTDTotal class.
The application code can also include instructs that use a set returned through the execution of an analytics expression directly in a relational database query to restrict relational data access based on aggregate analytics data. For example:
(Top10(SalesCube.Customer.CustomerID,
SalesCube.Salesamount,
Salescube.OrderDate.CurrentYear))
The analytics expression Top10 returns a set of ten customer ID's and associated sales amounts associated with the top ten sales amounts aggregated over the current year. The relational query Select statement is then performed on just the set of ten customer ID's to return just the customer ID's. This integration between the results returned by an analytics expression and a relational database query are possible because the analytics expression is strongly typed as either a set or a metric. As such, a reference to an analytics expression class will be considered valid by the compiler when it compiles the relational database query.
When an analytics expression template is specified in isolation (without the context of a cube query) as above, the analytics query framework will auto-generate a cube query with a single axis to project the result set generated by resolving the expression template. This scenario can be enabled by integrating the relational and analytics query frameworks to recognize and resolve individual parts of an overall query.
The metrics or sets returned by the execution of an analytics expression can also be integrated directly into another analytics expression. For example, in:
Top10(SalesCube.Customer.CustomerID, YearAgoDifference(SalesCube.SalesAmount, SalesCube.OrderDate.CurrentYear))
Top10 and YearAgoDifference are both analytics expressions defined by object classes as described above. YearAgoDifference provides a metric that measures the difference between SalesAmounts for the current year and the preceding year. This metric is then used to select the customer ID's of the ten customers with the largest gain in SalesAmount over the last year. Once again, because the analytics expression itself is typed, the compiler will consider the direct embedding of an analytics expression valid.
Although the subject matter has been described in language specific to structural features and/or methodological acts, it is to be understood that the subject matter defined in the appended claims is not necessarily limited to the specific features or acts described above. Rather, the specific features and acts described above are disclosed as example forms of implementing the claims.