LabKeyModules
LabKey Modules for ImmuneSpace. These modules form the basis of the data processing, analysis and visualization tools within ImmuneSpace. They depend on a number of related projects, mostly R packages but also packages set up specifically for testing or frontend development. All ImmuneSpace packages are hosted on GitHub and listed with their build status in https://github.com/RGLab/status#immunespace.
This repository includes many different modules. Thus, documentation for individual modules is included in a README in each module's directory.
Contributing
Development Process
LabKeyModules develompent is closely tied to the ImmuneSpace servers as well as the UITesting repository and depends on many other packages. The LabKeyModules Update Template outlines the entire process. Briefly:
- Identify an issue or new feature to add
- Create a feature branch off of
dev
namedfb_<description>
. Any changes in related packages should be made on a feature branch of he same name. - Develop on your local ImmuneSpace instance
- Create a feature branch off the UITesting
dev
branch matching the name of the LKM feature branch and update tests so that it passes on your local - Submit PRs to both LKM and UITesting and any other packages with related updates and request a review
- For the PR to be accepted, it must meet coding guidelines outlined below, and pass tests on the TEST server.
- Merge PRs into
dev
branch and submitdev
->main
PRs. dev
->main
PRs must pass all tests and go through code review- Install updates on PROD server.
Code Structure
- Each module gets its own directory. Each module directory should include a README, based on README_TEMPLATE.md, outlining the basic functionality, dependencies, any other special considerations, and linking to relevant documentation online or in Notion.
- For more details, refer to LabKey's documentation on developing modules.
- For React-based modules, follow guidelines outlined in immunespace-frontend-tools
Testing
Any changes should be tested. A PR without corresponding tests will not be accepted. Add integration tests to UITesting and cross-link PRs. Unit test React-based modules using Jest.
Documenting
If you are creating a new module, be sure to include a README in the module directory, following the README_TEMPLATE.md format. When making changes to a module, be sure to update the README file for that module. If a module requires additional documentation, add a Notion page.
Declaring dependencies
Any dependencies on other modules should be specified in the local build.gradle file. More documentation on dependency declaration. These, as well as any external dependencies should also be summarized in the README.
Versioning
Each module has its own version number, specified in the SchemaVersion
property of the module.properties file. LabKey uses module versioning for some business logic like deciding when to run a SQL script so you may need to update SchemaVersion
to get your module to work. At the very least, the minor version number should be updated when updates to a module get merged into dev.
Style Guide
Language-Agnostic guidelines
LabKeyModules is built on many different coding languages, including but not limited to R, SQL, xml, Javascript, and Typescript. With some exceptions, we follow LabKey's coding guidelines, and follow some language-agnostic guidelines:
- Use descriptive variable names. See language-specific guidelines around casing. Avoid abbreviations or look up shorthands in the glossary of variable abbreviations.
- Include descriptive code comments. Comment every functions and method. If your code is well-written it should be clear what it is doing so comments should explain the why, not the what.
R
Please refer to the R style guide described in ImmuneSpaceR for guidelines on R code. Run styler::style_file()
on any R or Rmd files you edit for automatic style enforcement.
SQL
LabKey summarizes SQL style guidelines in their Coding Guidelines page, copied below:
-
Use singular nouns for table, view, and query names.
-
Capitalize keywords.
-
Explicitly name all constraints. If you do not, they are assigned an auto-generated default name, which makes them much harder to manipulate later.
- Primary keys: PK_TableName
- Foreign keys: FK_TableName_ColumnName
- Indices: IDX_TableName_ColumnNames
- Unique: UQ_TableName_ColumnNames
-
When there is no “natural” key for a table, use the table’s name as part of the key column name. For example, use “CustomerID” as the Customer table’s PK instead of “RowID”.
-
GUID columns should be named with a “GUID” suffix. For example, “CustomerGUID”.
-
Junction table names should be the concatenation of the two linked table names. For example, “AppointmentProvider” is a junction table connecting “Appointment” and “Provider”.
xml
We generally do not create our own xml document formats and are pretty limited by LabKey's formats. Follow format from examples in LabKey documentation.
html
Refer to google's html style guide:
- indent by 2 spaces (don't mix tabs and spaces!)
- use only lower-case when possible
css
- First check to see if there is already a class which will accomplish your purpose or that you can extend
- All new class names should be lower case, and use dashes as separators. If they are specific to a module, the class name should start with the module name eg
datafinder-button
. If they are shared between modules, it should start withimmunespace-
egimmunespace-button
.
TypeScript and React
Our React-based modules are based on LabKey's guidelines and modeled on LabKey's demo module. Refer to immunespace-frontend-tools for the React style guide, and the Notion page for full documentation on how to develop React modules.
Helpful Documentation and References
SQL
LabKey has their own flavor of SQL, so refer to LabKey SQL reference when writing SQL scripts. Refer to LabKey's SQL script conventions for guidelines on developing SQL scripts.
ETL
We use LabKey ETLs in several different modules for tasks such as loading, caching, or updating data. See the LabKey ETL documentation for details on how to set up an ETL. An existing ETL can be run from the DataIntegration module.