/psi-header

Configurable file header and changes tracking extension for VSCode

Primary LanguageTypeScriptMIT LicenseMIT

:: Table of Contents ::

Overview

The psioniq File Header VSCode Extension will insert a header into the current document - either at the start of the document or at the current cursor position. The header can be configured globally and/or per language. However, the configuration separates the comment syntax from the template body so it is likely that a single template will be able to cover most languages.

It can optionally log modifications to the file via the change tracking feature which will update the header whenever the file is saved.

It is also possible to add templated comment blocks to the header to record for example a historic summary of changes to the file.

There are a veritable plethora of configuration options to cover your innermost headular cravings.

To report bugs, issues, suggestions use the github repository.

Here is a sample output:

/*
 * File: \Users\me\Development\psioniq\myProject\src\myPrecious.js
 * Project: \Users\me\Development\psioniq\myProject
 * Created Date: Saturday December 31 2016
 * Author: Arthur Bodkin, esq
 * -----
 * Last Modified: Sunday January 01 2017
 * Modified By: Tammy Bodkin
 * -----
 * Copyright (c) 2016 psioniq Global Enterprises, Inc
 */

To add a new header manually:

  • thump F1 and type Header Insert; or
  • type the keyboard shortcut ctrl-alt-H then ctrl-alt-H.

To insert an entry into the change log:

  • hit F1 and type Header Change Log Insert; or
  • type the keyboard shortcut ctrl-alt-C ctrl-alt-C.

Once inserted, the cursor will be placed at the end of the new log entry.

Now grab a coffee and read on...

Summary of Features

Refer to Configuration for the various extension settings details.

  • Adds a generic or language-specific header at the current cursor location or at the top of the file.
  • Optionally track changes in the header each time the file is saved.
  • Configurable whitelists and blacklists to determine which files can be change tracked.
  • Can automatically add a header to new files.
  • Compatible with VSCode Multi Root Workspaces.
  • Separates language specific elements (e.g. comment block begin and end) from the template body to minimise the number of templates you might need to manage.
  • Configuration option to force the header to the top of the document - overridable per language.
  • Configuration option to add additional blank lines after the header - overridable per language.
  • Configurable options to add text before or after the header (e.g. pre-processor commands).
  • Provides a default template body for all languages out of the box.
  • Provides language-specific syntax settings (e.g. comment block syntax) out of the box.
  • Configure your own custom language syntax globally and/or per language.
  • Create your own custom templates - global and/or per language.
  • Map custom template and syntax settings across languages so you can easily reuse them.
  • Provides case-insensitive System Variables for placeholder value substitution.
  • Provides case-sensitive System Functions for configurable placeholder value substitution.
  • Allows the overriding of system variable values with static global values within the configuration.
  • Create an unlimited number of custom static variables for use throughout your custom templates.
  • Header insertion can be run manually via the key shortcut ctrl+alt+H then ctrl+alt+H.
  • Can automatically insert license text based on SPDX license IDs or from a local text file.
  • Allows changes logging for recording a history of changes to the file.

Commands

This extension adds the following commands to VSCode:

Command name Keyboard Shortcut Description
Header Insert ctrl-alt-H ctrl-alt-H Inserts a new file header
Header Change Log Insert ctrl-alt-C ctrl-alt-C Inserts a new change log entry into an existing header

System Variables

The following system variables are available for placeholder substitution in your templates. The variable names are case-insensitive.

Variable Name Description
date The current date using the current locale (also see the dateformat() system function below for a formattable date string version).
time The current time using the current locale.
year The current year.
filepath The fully-qualified name of the file.
fullpath The fully-qualified path to the file (excludes the filename).
filerelativepath The file name including the relative path within the project.
relativepath The relative path to the file within the project (excludes the filename).
filename Just the file name without the path details.
filenamebase Just the file name without the path details or extension.
projectpath The fully-qualified path to the root directory of the project.
projectname Attempts to read package.json (in the current or any parent directory) for either a displayName or name property. If there is no package.json file and the file has been saved to disk, it will return the project path's base name.
projectslug Provides a version of projectname for use in url's and links. E.g. "@davidquinn/psi-header" would return "davidquinn/psi-header"
projectversion Attempts to read package.json (in the current or any parent directory) for a version property.
company The name of your company. In this release it defaults to "Your Company".
author Will attempt to get the name of the current user. The options are coalesced in the following order: psi-header.config.author then fullname from the OS then username from the OS. If none of these return a value, it defaults to "You".
initials Your initials (where you don't want the whole author name
authoremail The email address of the file author. In this release it defaults to "you@you.you".
licensetext The full text of the license. This is only required if psi-header.config.license option is set to "Custom".
copyrightholder Used in some licenses. If not provided it defaults to the same value as company.
licensename The name of the license. If not using a custom license, this is determined automatically.
licenseurl The url for the license. If not using a license, this is determined automatically.
spdxid The SPDX License ID for the license. If not using a custom license, this is determined automatically.
hostname The value of the psi-header.config.hostname setting if set, otherwise the local machine's hostname as provided by the OS.

Within your template, surround the variable names with two sets of chevrons (e.g. ``<>. Also, you can modify the text case of the output for any of the system variables by appending !U` or `!L` to the end of the variable name (e.g. `<<filepath!U>>` to render the file path in uppercase or `<<filepath!L>>` for lower case).

You can also create your own static custom variables (for example if you are using this extension within a team or you need project-specific variables in your template) by adding your own variables to psi-header.variables then referring to them within your template like the following example which adds a custom variable called projectCreationYear:

	"psi-header.variables": [
		["projectCreationYear", "2017"]
	],
	"psi-header.templates": [
		{
			"language": "*",
			"template": [
				"File: <<filename>>",
				"Project: <<projectname>>",
				"File Created: <<filecreated('dddd, Do MMMM YYYY h:mm:ss a')>>",
				"Author: <<author>> (<<authoremail>>)",
				"-----",
				"Last Modified: <<dateformat('dddd, Do MMMM YYYY h:mm:ss a')>>",
				"Modified By: <<author>> (<<authoremail>>>)",
				"-----",
				"Copyright <<projectCreationYear>> - <<year>> <<copyrightholder>>, <<company>>"
			]
		}
	]

NOTE: You can include system variables in the beforeHeader and afterHeader properties of your language configurations. However, they will only be processed when the header is first added to your file and will not be updated on subsequent file saves. Why? Because the extension has no idea what may have changed or been edited outside of the template since it was added to the file, and possibly the worst thing that the extension could do is delete or mess up your carefully crafted source code by assuming that the structure of the before and after blocks in your language template exactly matches the structure that existed at the time the header was added.

System Functions

The following case-sensitive system functions are available for configurable placeholder value substitution. These are similar to System Variables but they take arguments and they do not support the text case modifiers.

Function Name Description
dateformat(format) The current date or date part using format strings. This function takes a single string argument which represents the moment.js compatible format string.
filecreated(format) The file created date and time using format strings. This function takes a single string argument which represents the moment.js compatible format string (surrounded in single or double quotes). It can also be called without arguments to use the current locale date format. If the file created date cannot be determined it will return the current date and time (usually because the file has not yet been saved to disk, or the operating system failed to return the creation date and time). Refer to this known issue for potential issues with some Linux setups.
yeartoyear(from, to) Generates a string in the form "YYYY - YYYY" or just "YYYY" of both dates evaluate to the same year. Useful for copyright messages. Refer to year to year for details.
padline(char, maxLength) Pads the end of the line with the char up to the maxLength characters. Refer to padline for details.

As with system variables the functions can appear in the beforeHeader and afterHeader properties of your language configuration, with the same constraints around updating.

Date Formats in System Functions

filecreated can also return the current locale date string by passing no arguments. Both the following will work:

	<<filecreated()>>
	<<filecreated>>

The following examples would both output a date like 2017-04-14

    <<dateformat('YYYY-MM-DD')>>
    <<filecreated('YYYY-MM-DD')>>

And the following would both generate something like Friday, April 14th 2017, 8:50:19 am

	<<dateformat('dddd, MMMM Do YYYY, h:mm:ss a')>>
	<<filecreated('dddd, MMMM Do YYYY, h:mm:ss a')>>

These functions use Moment.js and can use all Moment.js format string options.

yeartoyear

Generates a string in the form "YYYY - YYYY" or just "YYYY" if both dates evaluate to the same year. Useful for copyright messages.

This function takes 2 arguments: from determines the first date and to determines the second date. Possible values for each argument are shown in the following table. If both evaluate to the same year only a single string ("YYYY") is returned.

Property Value Description
fc The file creation date. This is the default for the from date. Not case-sensitive.
now The current year. This is the default for the to date. Not case sensitive.
var:name Use a static variable where name is the name of the static variable.
any other string String written as is. Useful for setting a static year. Do not include any commas in the string.

Additionally, the flag "!P" can be appended to an argument above to request reuse of the previous year value in an existing header. When "!P" is specified and Changes Tracking is updating an existing header, the function will attempt to reuse the existing from or to year. The function looks at the position of the yeartoyear call in the line being replaced for format YYYY - YYYY (spaces optional) or else YYYY. When writing a new header, or when the required year format is not matched, the "!P" flag is ignored. A particularly useful case is yeartoyear(fc!P, now) to use file creation date for a new header, and then always keep the same from date no matter how the file migrates filesystems, version control systems, etc.

If only one argument is passed in, it is assumed to be the from date and to will use its default. If no arguments are passed then in both will use their default values. If no arguments are passed in it will work even without the function brackets. The arguments can optionally include double or single quote qualifiers, but will also work without any qualifiers.

Valid examples:

<<yeartoyear>>
<<yeartoyear()>>
<<yeartoyear(2020)>>
<<yeartoyear(fc)>>
<<yeartoyear(fc!P)>>
<<yeartoyear(2020, 2021)>>
<<yeartoyear(1976,now)>>
<<yeartoyear(fc,2020)>>
<<yeartoyear(fc,now)>>
<<yeartoyear("fc","now")>>
<<yeartoyear('fc!P','9999!P')>>
<<yeartoyear(fc!P,now)>>
<<yeartoyear(i am a rabbit, 2020)>>

padline

This function pads the end of a single line with the char up to the maxLength characters. Only the first character in char will be used - any subsequent characters will be ignored. If maxLength is not defined, the line will be padded up to the length of psi-header.lang-config.lineLength (which itself defaults to 80). Regardless, this function will never cause the line to have more characters than defined by lineLength.

Padding will always be applied to the end of the line, regardless of where in the line the function is declared.

Valid examples:

<<padline('-')>> // this will add the dash character up to the line length defined for the language.
<<padline('+', 5)>> // this will add up to 5 plus characters toi the end of the line.

Configuration

It is quite possible to use this extension without making any changes to your VSCode's settings (although you probably want to set up a couple of variable values like author and company at least). Extensive configuration options are available should you wish to get your hands dirty.

Following is an example settings file with every conceivable psi-header option (the rest of this section provides the detail for each of the settings).

WARNING: Do not just paste this example into your settings.json file because there will be conflicting values - the example is just provided to show how the settings file entries are structured.

{
	"psi-header.config": {
		"forceToTop": true,
		"blankLinesAfter": 1,
		"spacesBetweenYears": false,
		"license": "MIT",
		"author": "Arthur Bodkin",
		"initials": "AB",
		"authorEmail": "arthur@bodkin-enterprises.com",
		"company": "Bodkin World Domination Enterprises",
		"copyrightHolder": "Bodkin & Bodkin Ltd",
		"creationDateZero": "asIs",
		"hostname": "myhostname"
	},
	"psi-header.changes-tracking": {
		"isActive": true,
		"modAuthor": "Modified By:",
		"modDate": "Last Modified:",
		"modDateFormat": "dd/MM/yyyy hh:nn:ss",
		"include": [],
		"includeGlob": [],
		"exclude": ["markdown", "json", "jsonc", "shellscript"],
		"excludeGlob": ["./**/*/ignoreme.*"],
		"autoHeader": "autoSave",
		"enforceHeader": true,
		"replace": [
			"Filename:",
			"Project"
		],
		"updateLicenseVariables": false
	},
	"psi-header.variables": [
		["manager", "Old Mother Bodkin"],
		["projectCreationYear", "2019"]
	],
	"psi-header.lang-config": [
		{
			"language": "javascript",
			"begin": "/*",
			"prefix": " * ",
			"suffix": " *",
			"lineLength": 80,
			"end": " */",
			"forceToTop": true,
			"blankLinesAfter": 3,
			"beforeHeader": [],
			"afterHeader": [],
			"rootDirFileName": "package.json",
			"modAuthor": "Modified By:",
			"modDate": "Last Modified:",
			"modDateFormat": "dd/MM/yyyy hh:nn:ss",
			"replace": [
				"Filename:",
				"Project"
			],
			"ignoreLines": []
		},
		{
			"language": "typescript",
			"mapTo": "javascript"
		}
	],
	"psi-header.templates": [
		{
			"language": "javascript",
			"template": [
				"File: <<filename>>",
 				"Project: <<projectname>>",
				"Created Date: <<filecreated('dd MMM yyyy')>>",
				"Author: <<author>",
				"-----",
				"Last Modified: <<date>>",
				"Modified By: <<author>>",
				"-----",
				"Copyright (c) <<yeartoyear(fc, now)>> <<company>>",
				"-----",
				"HISTORY:",
				"Date      \tBy\tComments",
				"----------\t---\t---------------------------------------------------------"
			],
			"changeLogCaption": "HISTORY",
			"changeLogHeaderLineCount": 2,
			"changeLogEntryTemplate": [
				"",
				"<<dateformat('DD-MM-YYYY')>>\t<<initials>>\t"
			],
			"changeLogNaturalOrder": false,
			"changeLogFooterLineCount": 0
		},
		{
			"language": "typescript",
			"mapTo": "javascript"
		}
	],
	"psi-header.license-text": [
		"This will never show because it is only relevant if",
		"psi-header.config.license equals Custom"
	],
	"psi-header.license-reference": {
		"uri": "path/to/license.file",
		"uriIsLocalFile": true
	}
}

There are some specific settings that you must setup if you want to use the Change Log feature.

Settings can be added as User and/or Workspace and/or WorkspaceFolder settings - VSCode handles the majik of merging them together. Workspace Folder settings take precedence over Workspace settings which take precedence over User settings (which in turn take precedence over Default values).

When generating a header, the extension will do the following for the language-specific settings (psi-header.lang-config and psi-header.templates):

  1. Start with either the configured or the built in defaults.
  2. If there is a configuration setting that matches the document language, options set there will overwrite those from the default; else
  3. If there is a global configuration (language = "*"), options set there will overwrite those from the default.

Intellisense is provided for the options within the user and workspace settings files in VSCode.

The configuration settings are organised into the following areas described below:

  • psi-header:
    • config: global configuration settings;
    • changes-tracking: settings relates to changes tracking;
    • variables: an array of key/value pairs for variable substitution values;
    • lang-config: an array of language-specific (and general) settings (comment styles, etc);
    • templates: an array of language-specific (and general) header templates;
    • license-text: an array of strings that defines custom license text.
    • license-reference: options for reading the license from a local file or a web link.

Global Options

Options that affect the whole extension. In some cases these defaults can be overridden within specific language configurations.

Configuration Section: psi-header.config.

Option Description
forceToTop If true, it will ignore the current cursor position and insert the header at the top of the document. If false (the default), the header will be inserted at the current cursor position. Can be overridden for specific languages (via psi-header.lang-config).
blankLinesAfter Specify how many blank lines to insert after the header comment block. Default is 0 (zero). Can be overridden for specific languages (via psi-header.lang-config).
spacesBetweenYears If true, include spaces between years ("YYYY - YYYY"), otherwise omit them ("YYYY-YYYY"). Default is true.
license The SPDX License ID of the license to insert into the header (or "Custom" or "CustomUri" if providing your own license text). Refer to License Information for details.
author Your name - used by the author system variable. Optional with no default.
initials Your initials - used by the initials system variable. Optional with no default.
authorEmail Your email address - used by the authoremail system variable. Optional with no default.
company Your Company's name - used by the company system variable. Optional with no default.
copyrightHolder Your copyright name - used by the copyrightholder system variable. Optional with no default.
creationDateZero What to do when the operating system returns Epoch Zero (1 Jan 1970) for the file creation time (e.g. when the source file is an NTFS-mounted file in Linux). Valid options are:
- 'asIs': (default) just write the date provided by the OS;
- 'blank': do not write a creation date;
- 'now': write creation date as now.
hostname Overrides the local machine hostname returned by the OS with a custom value when using the hostname system variable.
ignoreAuthorFullName On some systems it can be expensive to attempt to get fullname from the OS, so this option ignores this when determining the author variable value

Changes Tracking Configuration

Options that affect changes tracking.

Configuration Section: psi-header.changes-tracking.

Option Description
isActive If true, will activate changes tracking which will analyse every file during save. Default value is false.
modAuthor Identifies the label used on the comment line where the modified by value is shown. Default value is "Modified By:".
modDate Identifies the label used on the comment line where the date modified value is shown. Default value is "Last Modified:".
modDateFormat The format string for the modified date value. Valid values are either "date" (system date - same as the date system variable) or a Moment.js format string. The default value is "date". Note that this setting is ignored if modDate line is based on a custom string.
include Defines an array of VSC language IDs for the file types to include in changes tracking. The default is an empty array which indicates any file type.
includeGlob Defines an array of file globs for the files to include in changes tracking. The default is an empty array which indicates any file.
exclude Defines an array of VSC language IDs for the file types to exclude from changes tracking. The default is an empty array which indicates no exclusions.
excludeGlob Defines an array of file globs for the files to exclude from changes tracking. The default is an empty array which indicates no exclusions.
autoHeader Determines whether the header should be added automatically to new files. Refer to the Auto Header section for details.
enforceHeader Determines if the extension should automatically check the file and add a header whenever the file is saved if an existing header is not found. This setting is independent of autoHeader and works best with psi-header.config.forceToTop set to true. Refer to the Enforce Header section for details and usage recommendations.
replace An array of template line prefixes that define additional header lines to replace during a file save. By way of example, you could use this to ensure that changes to file name or project name are always updated during save (it happens!).
updateLicenseVariables This option determines if license variables will be updated during a changes tracking save. Set this to true if your configuration would cause updates to header lines that include any license variables. Otherwise leave if off because it is a fairly expensive operation to process the SPDX license data every time a file is saved.

Variable Values

An array of name/value pairs that provide value substitution within templates. This can be used to override the system variables as well as add new items for your own custom templates.

Configuration Section: psi-header.variables.

Language Configuration

An array of objects that allow language-specific adjustments to be made to the configuration. You can provide a subset of values if you only want to override some of the settings. Each object can include the following options.

Configuration Section: psi-header.lang-config.

Option Description
language Mandatory. Either a file extension including the leading period; the VSCode Language ID; or "*" or "*DEFAULTS*" for default language configurations.
mapTo Optional. If provided, this language will use the specified language's configuration (and the settings below will be ignored). The value is a VSCode language ID. You can not mapTo a language that itself has the mapTo value set. Ignored if language = "*".
begin Optional - defaults to "/*". Determines the comment block opening text. This will be inserted as a line before the first line of the template. Refer to Compact Mode for information on headings with no begin and end lines.
prefix Optional - defaults to " * ". Determines a prefix for each body line of the header.
suffix Optional character(s) to form a line end suffix if you want block-style comment headers.
lineLength Optional - sets the default line length for block-style comment headers.
end Optional - defaults to " */". Determines the comment block closing text. This will be inserted after the last line of the template. Refer to Compact Mode for information on headings with no begin and end lines.
forceToTop Optional. Same as psi-header.config.forceToTop but just for this language. If set, this overrides the global setting.
blankLinesAfter Optional. Same as psi-header.config.blankLinesAfter but just for this language. If set, this overrides the global setting.
beforeHeader Optional. Allows multiple lines of text to be inserted before the beginning of the header comment block (e.g. pre-processor commands). NOTE: The extenion will not add comment prefixes to this text, so you will need to include them in your text if necessary. You can add system variables to this section that are processed on initial header creation (but not re-processed on subsequent saves).
afterHeader Optional. Allows multiple lines of text to be inserted after the end of the header comment block (e.g. pre-processor commands). This will appear after any configured blankLinesAfter. NOTE: The extenion will not add comment prefixes to this text, so you will need to include them in your text if necessary. You can add system variables to this section that are processed on initial header creation (but not re-processed on subsequent saves).
rootDirFileName Optional. By default this extension looks for a file called package.json to determine the project's root path. This allows you to also search for an additional file name that should exist in the root path. The contents of the file are irrelevant (it can be a blank file). Only really needed if you can't have a package.json file in your root path.
modAuthor Optional. Overrides the changes-tracking.modAuthor setting for this language. Identifies the label used on the comment line where the modified by value is shown. There is no default value as by default the changes tracking setting will be used.
modDate Optional. Overrides the changes-tracking.modDate setting for this language. Identifies the label used on the comment line where the date modified value is shown. There is no default value as by default the changes tracking setting will be used.
modDateFormat Optional. Overrides the changes-tracking.modDateFormat setting for this language. The format string for the modified date value. Valid values are either "date" (system date - same as the date system variable) or a Moment.js format string. There is no default value as by default the changes tracking setting will be used. Note that this setting is ignored if modDate line is based on a custom string.
replace Optional. Overrides the changes-tracking.replace setting for this language. An array of template line prefixes that define additional header lines to replace during a file save. If defined, the value here replaces the replace settings in changes-tracking for this language.
ignoreLines Optional array of strings. Used by the logic that determines if a header needs to be auto-inserted to exclude any lines that start with the specified string(s) that may appear before the header. Useful where VSCode or another extension may try to insert lines above your header. Refer to the Auto Header section for more information.

Default Language Configurations

The extension includes 2 optional special lang-config entries:

  • "language": "*": the configuration to use when there is no specific lang-config entry for the source file; and
  • "language": "*DEFAULTS*": default optional property values to use for any property that is not specifically set in the chosen lang-config (which might be "*"). This option was introduced in v1.22.0, and the extension will default to the previous behaviour of {"begin": "/*", "prefix": " *", "end": " */"} if this entry is not present.

BE CAREFUL!: The "*DEFAULTS*" entry overrides the previous system default values. So it may affect headers for languages you currently use if you add this to an existing VSCode setup. Unless you have a burning desire to change the system defaults, you probably do not need this configuration.

Templates

An array of template definitions. Each definition must include either mapTo or template. Includes the following options.

Configuration Section: psi-header.templates

Option Description
language Mandatory. Either a file extension including the leading period; the VSCode Language ID; or "*" or "*DEFAULTS*" for default template configurations.
mapTo Optional. If provided, this language will use the specified language's template (and will ignore the following template value). The value is a VSCode language ID. You can not mapTo a language that itself has the mapTo value set. Ignored if _language = ""_*.
template This must be provided if mapTo is not declared of if "language": "*DEFAULTS*". Includes an array of strings that represent the body of the header. No need to include the comment block syntax.
changeLogCaption Used by the Change Log feature. Defines the caption for the change log that must also appear in the main header template. The extension uses this caption to work out where to place a new change log entry. Ignored if changeLogNaturalOrder is true.
changeLogHeaderLineCount Used in the Change Log feature to define the number of lines in the main template between the above changeLogCaption and the log entries. This can be used to configure the main template to include column headings for the change log. Defaults to 0 if not provided. Ignored if changeLogNaturalOrder is true.
changeLogEntryTemplate The template for a change log entry. Allows overriding of the default change log entry template.
changeLogNaturalOrder If true, change log entries will be shown in chronological order (latest entry last) at the end of the header. The default is false.
changeLogFooterLineCount Used in the Change Log feature to define the number of lines in the main template between the bottom of the footer and the log entries. Defaults to 0 if not provided. Ignored if changeLogNaturalOrder is false.

NOTE: Also, mapTo is ignored if the language value is set to "*".

Default Template Configurations

The extension includes 2 optional special templates entries:

  • "language": "*": the template configuration to use when there is no specific templates entry for the source file; and
  • "language": "*DEFAULTS*": the default optional property values to use for any property that is not specifically set in the chosen templates (which might be "*"). This option was introduced in v1.22.0, and the extension will default to the previous behaviour which defines a basic template if this entry is not present.

BE CAREFUL!: The "*DEFAULTS*" entry overrides the previous system default values. So it may affect headers for languages you currently use if you add this to an existing VSCode setup. Unless you have a burning desire to change the system defaults, you probably do not need this configuration.

License Text

An optional array of strings for defining custom license text. Used where psi-header.config.license = "Custom".

Configuration Section: psi-header.license-text

License Reference

Options for reading the license text from a local file or for setting the license URL. Requires psi-header.config.license to be set to "CustomUri". Refer to License Information for details.

Configuration Section: psi-header.license-reference

Option Description
uri Either (a) a fully qualified file name including absolute path information, or (b) a filename with no path or (c) a URL.
uriIsLocalFile If true, the contents of the file at the uri will be added to the <> variable in the header. Otherwise the uri will be treated as a web link and assumed to be complete.

Compact Mode

The header can be created in "compact mode" - that is, without begin and end lines on the comment block. To activate this, you must set both the lang-config.begin and lang-config.end configuration values to an empty string for any language where you want this behaviour. You can do this on the default language configuration (lang-config.language = "*") to apply to all languages. Obviously this will cause the header to work like a series of single line comments, so you must also make sure that the lang-config.prefix property is set to a valid single line comment prefix.

Also note that the change tracking will not auto-insert a header in compact mode if there are any lines in the file that start with the lang-config.prefix.

Again, this will only work if both the lang-config.begin and lang-config.end configuration values are present and set to an empty string.

Example language configuration:

	"psi-header.lang-config": [
		{
			"language": "javascript",
			"begin": "",
			"end": "",
			"prefix": "// "
		}
	]

Block-Style Comment Headers

You can create block-style comment headers using the psi-header.lang-config's suffix setting, and optionally the lineLength setting. This can be used to provide a header in the following format:

 ###############################################################################
 # File: \Users\me\Development\psioniq\myProject\src\myPrecious.xyz            #
 # Project: \Users\me\Development\psioniq\myProject                            #
 # Created Date: Saturday December 31 2016                                     #
 # Author: Arthur Bodkin, esq                                                  #
 # -----                                                                       #
 # Last Modified: Sunday January 01 2017                                       #
 # Modified By: Tammy Bodkin                                                   #
 # -----                                                                       #
 # Copyright (c) 2016 psioniq Global Enterprises, Inc                          #
 ###############################################################################

To use block-style just add a non-blank suffix property to any language where you want this feature (or add it to the default '*' language configuration). The suffix must be one or more characters long in order for this to work.

You can also influence how long each line is by setting the lineLength property for the language. If this is not set, the extension will attempt to use the current editor value (wordWrapColumn) or will otherwise default to 80.

When working out the line length, the extension will take account of any tab characters in the line.

The extension will not attempt to wrap lines that are already longer than the line length. It also does not add the suffix to the beginning and end lines or to change history lines.

A Note about Project Paths

When this extension was originally written VSCode only supported opening a single directory in a workspace. So, working out the root directory was reasonably simple - life was good! However, now with Multi Root Workspaces we can no longer assume the root directory (infact Microsoft has deprecated the method that returned the root directory).

Therefore, placeholders that need to know the project root directory (filerelativepath, projectpath and projectname) try to work it out by iterating up the directory structure (starting at the current editor file location) until they come to a package.json file or a filename set in the psi-header.lang-config.rootDirFileName configuration setting. If either one is found then its location is assumed to be the root - otherwise it just assumes the same directory as the edited file.

License Information

The psi-header.config.license setting expects either a valid SPDX license ID or one of "Custom" or "CustomUri" if you are providing your own license text. The settings are used to populate the following system variables: <<licensetext>>, <<licensename>>, <<licenseurl>> and <<spdixid>>.

"Custom"

When set to "Custom", you need to provide the license text via the psi-header.license-text setting. This will populate the <<licensetext>> system variable.

"CustomUri"

When set to "CustomUri" you need to provide further details in the psi-header.license-reference section. The License reference setting uri should be either:

  1. a local filename including an absolute path to point to a file in a specific location; or
  2. a filename without a path where the file exists somewhere on the same directory branch as the file being edited. The extension will search from the edit file location up the directory structure until it finds a file with the correct name; or
  3. a URL - note that the extension will not check the validity of the entered URL.

If uriIsLocalFile is false, it will just use the uri to populate the <> system variable.

If uriIsLocalFile is true, the extension will first test if the uri is valid as an absolute path (in which case option 1 is used) and if not it will ignore any path information in the uri and just use the basename with option 2. Assuming it finda a valid text file, it will attempt to copy the contents into the <<licensetext>> system variable in your template.

SPDX

Assuming you provide a valid SPDX ID, you can populate the following system variables in your template:

  • <<spdxid>> system variable wil ladd the SPDX ID to your header.
  • <<licensename>> is the name of the license provided by SPDX.
  • <<licenseurl>> is the URL for the license provided by SPDX.
  • <<licensetext>> is the license text provided by SPDX.

The extension does some clean up of the SPDX license text (mapping to variables, etc) but not everything is cleaned. In particular, a number of licenses use a placeholder logic based on <<var;...>> that this extension does not try to convert - and some licenses have placeholder text like <insert your slartibartfast here. We wore an onion on our belt because that was the fashion of the day>. If you find hokey little anomolies that can be dealt with, let me know. Otherwise, I suggest you copy the license text into your custom license settings or local text file and fix it there.

Refreshing the License Text System Variable

By default, license variables are only processed on initial header creation. This is because processing the SPDX license data is an expensive operation to do on every file save. If you use changes tracking and if your license variables are not being correctly processed on file save, you will need to set the psi-header.changes-tracking.updateLicenseVariables option to true to have the text refreshed on every save.

Changes Tracking

This extension can optionally track changes to files during save by writing the last modified date and/or user to the header comment block. It does this by looking for specific text (ignoring initial whitespace) at the start of individual lines within the header, and replacing the whole line. It will only search the first multi-line comment block within each file.

It works when saving either single or multiple files (e.g. during a Save All). It will only update the details if VSC reports the document as "dirty".

By default change tracking processes every dirty file on save. You can restrict which files are processed via the psi-header.changes-tracking properties:

  • include: an array that provides a whitelist of language file types to include;
  • exclude: an array that provides a blacklist of language file types to exclude;
  • includeGlob: an array that provides a whitelist of file globs to include;
  • excludeGlob: an array that provides a blacklist of file globs to exclude.

The extension uses the following logic to work out whether to change-track the current file:

  • if both include and includeGlob are null or empty then any file can be included. However, if either (or both) include or includeGlob are not empty then only files that satisfy the non-empty whitelist(s) can be included; then
  • if the current file satisfies the above criteria, the extension will test to make sure the file is not covered by the exclude and excludeGlob settings.

By default changes tracking is disabled - you can activate it via the psi-header.changes-tracking.isActive boolean configuration property.

Early versions of the extension simply replaced everything after the line identifier. However, from version 1.1.3 you can use the template to customise the whole line. These two options are described below.

Option 1 Simple Replacement

It will look for lines within the header that start with languageCommentPrefix + trimStart(label) (e.g. " * Date Modified:" or " * Modified By:") and will replace the whole line with languageCommentPrefix + label + newValue. Where:

  • languageCommentPrefix is the comment line prefix for the document's language (psi-header.lang-config[language].prefix);
  • label is either:
    • the configured psi-header.changes-tracking.modAuthor (defaults to "Modified By:"); or
    • the configured psi-header.changes-tracking.modDate (defaults to "Date Modified:").
  • and newValue is either:
    • the author's name (same logic as the author system variable); or
    • the current date formatted via the configured psi-header.changes-tracking.modDateFormat (refer to the configuration settings for details).

Note that it will replace the whole line so is not suitable for lines where you want to control the text of the line.

Also, because the whole line is replaced, you need to make sure your label configuration includes all characters before the new value (e.g. the ":" in the above defaults). Although, the extension will insert a space before the value if necessary.

So, taking the following example template from the beginning of the README:

"psi-header.templates": [
	{
		"language": "*",
		"template": [
			"File: <<filepath>>",
			"Project: <<projectpath>>",
			"Created Date: <<filecreated('dddd MMMM Do YYYY')>>",
			"Author: <<author>>",
			"-----",
			"Last Modified:",
			"Modified By:",
			"-----",
			"Copyright (c) <<year>> <<company>>"
		]
	},
]

Let's say Uncle Jack Bodkin modified the file three days after Tammy, then (assuming default values) the header would look like the following after save:

/*
 * File: \Users\me\Development\psioniq\myProject\src\myPrecious.js
 * Project: \Users\me\Development\psioniq\myProject
 * Created Date: Saturday December 31 2016
 * Author: Arthur Bodkin, esq
 * -----
 * Last Modified: Tuesday January 03 2017
 * Modified By: Uncle Jack Bodkin
 * -----
 * Copyright (c) 2016 psioniq Global Enterprises, Inc
 */

NOTE: What is important in the above is that there is no text beyond the label in the template for the two lines we are replacing.

Option 2 Template Substitution

If there are any characters in the template on either of the comment lines after the label then the extension will preserve it during the update. You can include text as well as any of the system variables and functions.

You can also use this method to update other lines from the template via the psi-header.changes-tracking.replace array to configure it globally or via the psi-header.lang-config[].replace settings to override for a specific language. The array holds strings that represent the beginning of the line from the template (excluding the line prefix) - you just need to include enough characters from the beginning of the line to ensure uniqueness.

Note that the psi-header.changes-tracking.modDateFormat configuration setting is ignored when using this option.

So, modifying the "Last Modified:", "Modified By:", and "Copyright" lines in the template from the earlier example in Option 1,

"psi-header.templates": [
	{
		"language": "*",
		"template": [
			"File: <<filepath>>",
			"Project: <<projectpath>>",
			"Created Date: <<filecreated('dddd MMMM Do YYYY')>>",
			"Author: <<author>>",
			"-----",
			"Last Modified: <<filecreated('dddd MMMM Do YYYY h:mm:ss a')>>",
			"Modified By: the developer formerly known as <<author>> at <<<authoremail>>>",
			"-----",
			"Copyright (c) <<yeartoyear(fc!P,now)>> <<company>>"
		]
	}
],
"psi-header.changes-tracking": {
	...
	"replace": [ "Copyright" ],
	...
}

or...

"psi-header.templates": [
	{
		"language": "*",
		"template": [
			"File: <<filepath>>",
			"Project: <<projectpath>>",
			"Created Date: <<filecreated('dddd MMMM Do YYYY')>>",
			"Author: <<author>>",
			"-----",
			"Last Modified: <<filecreated('dddd MMMM Do YYYY h:mm:ss a')>>",
			"Modified By: the developer formerly known as <<author>> at <<<authoremail>>>",
			"-----",
			"Copyright (c) <<yeartoyear(fc!P,now)>> <<company>>"
		]
	}
],
"psi-header.lang-config": [
	{
		"language": "*",
		...
		"replace": [ "Copyright" ],
		...
	}
]

Because there is now text after the labels on the "Last Modified:" and "Modified By:" lines, the extra text is used to generate their output. The replace array specifies replacing the "Copyright" line, and the "!P" in the argument to the yeartoyear function specifies to preserve the initial year of the range.

/*
 * File: \Users\me\Development\psioniq\myProject\src\myPrecious.js
 * Project: \Users\me\Development\psioniq\myProject
 * Created Date: Saturday December 31 2016
 * Author: Arthur Bodkin, esq
 * -----
 * Last Modified: Tuesday January 03 2017 09:37:28 am
 * Modified By: the developer formerly known as Uncle Jack Bodkin at <uncle.jack@psioniq.net>
 * -----
 * Copyright (c) 2016 - 2017 psioniq Global Enterprises, Inc
 */

Auto Header

This extension can be configured to automatically add a file header on creating a new file via the psi-header.changes-tracking.autoHeader setting. The valid values for this setting are:

  • off: (default) headers will not be added to new files automatically;
  • manualSave: a header will be added but the file will not be automatically saved; or
  • autoSave: the header will be added and the file immediately saved.

If the file is added via the New File icon in the Explorer the header will be created immediately. However, if the file is created via the File->New File menu item or via Ctrl/Cmd-N the header will not be added until you perform a physical save. Why? Mainly because we do not know what language the file will use until it is saved. The extension will not add a header to a new file that already contains a multi-line comment block.

The auto header configuration will honour the include, exclude, includeGlob and excludeGlob settings under psi-header.changes-tracking.

Where the header is supposed to be at the top of the file, you may want to use the psi-header.lang-config.ignoreLines string array setting. The save process will exclude lines that start with any of the ignoreLines strings when trying to work out if the file already has a header. Use this to get around the problem where this extension is tricked into adding a duplicate header because VSCode has annoyingly auto-inserted import statements (etc) above your carefully constructed header. Note that blank lines at the top of the file are always ignored.

Enforce Header

The Auto Header setting will only create headers for new files added directly via VSCode. To insert a header in any file during save, set psi-header.changes-tracking.enforceHeader option to true. enforceHeader will scan the file during save to check if there is a header - it works best with psi-header.config.forceToTop set to true, otherwise any comment block in the file could be interpreted as a header.

If you are using this setting, I would recommend that you also add "**/settings.json" to psi-header.changes-tracking.excludeGlob to ensure that headers do not get added to VSCode's settings file. I also suggest you add ["jsonc","json"] to the psi-header.changes-tracking.exclude setting to ensure headers are not added to json files (jsonc is VSCode's language id for json files that are enabled for comments).

The psi-header.lang-config.ignoreLines discussion in the Auto Header section is particularly important to help ensure this setting sdoes not result in duplicate headers.

Change Log

This feature allows you to add change log entries to the header to record major changes to the current file. It provides a template for each change log entry and you then just add your own comments. By default it is configured to record the date and initials of the user to which you can add a short comment, but you can configure it to your needs.

By default, entries are added immediately after the Caption Line (described in the configuration section below) with the most recent entries at the top. Since v1.7.3, you can set the psi-header.templates[].changeLogNaturalOrder to true to have entries shown in natural chronological order (latest entry last).

To insert an entry into the change log, just hit ctrl-alt-C, ctrl-alt-C. Once inserted, the cursor will be placed at the end of the new log entry.

Note that the above call will fail if the template has not been correctly configured (see below) or if the document does not contain a header.

Configuring Change Logging

To configure this you must do one of either:

  1. add a caption line to your psi-header.templates[].template and reference the caption text in psi-header.templates[].changeLogCaption which must include enough of the caption line to enable it to be found within the header; or
  2. set psi-header.templates[].changeLogNaturalOrder to true.

It is not possible to use this feature without one of these settings.

Option #1 will show newest entries at the top of the log. You can also optionally add extra lines between the caption line and the change log entries via the psi-header.templates[].changeLogHeaderLineCount setting to add (for example) column headings for your entries. This setting records the number of lines in your template in between the caption line and the log entries - it excludes the caption line itself. This setting defaults to 0 (zero) - i.e. no extra lines.

Option #2 will show the entries in natural chronological order (last entry at the bottom) and appears at the end of the header. This option ignores the changeLogCaption setting and no caption line is required. You can also optionally add extra lines between the bottom of the header and the change log entries via the psi-header.templates[].changeLogFooterLineCount setting. This setting defaults to 0 (zero) - i.e. no extra lines.

Finally, the default log entry template is a single line with date then a TAB then initials then another TAB but you can create your own template via the psi-header.templates[].changeLogEntryTemplate setting - see examples below.

A simple example that just adds a caption to the header and uses the defaults for everything else is provided below:

"psi-header.templates": [
	{
		"language": "*",
		"template": [
			"File: <<filepath>>",
			"Project: <<projectpath>>",
			"Created Date: <<filecreated('dddd MMMM Do YYYY')>>",
			"Author: <<author>>",
			"-----",
			"Last Modified: <<filecreated('dddd MMMM Do YYYY h:mm:ss a')>>",
			"Modified By: the developer formerly known as <<author>> at <<<authoremail>>>",
			"-----",
			"Copyright (c) <<year>> <<company>>"
			"-----",
			"HISTORY:"
		],
		"changeLogCaption": "HISTORY:"
	},
]

would give output similar to the following:

/*
 * File: \Users\me\Development\psioniq\myProject\src\myPrecious.js
 * Project: \Users\me\Development\psioniq\myProject
 * Created Date: Saturday December 31 2016
 * Author: Arthur Bodkin, esq
 * -----
 * Last Modified: Tuesday January 03 2017 09:37:28 am
 * Modified By: the developer formerly known as Uncle Jack Bodkin at <uncle.jack@psioniq.net>
 * -----
 * Copyright (c) 2016 psioniq Global Enterprises, Inc
 * -----
 * HISTORY:
 * 2018-07-14	JB	Added a rabbit
 * 2018-05-12	AB	Fixed the type of bug that only a mother could love.
 */

Or for an example with a heading and custom template that uses a different date format and adds a blank line before each entry:

"psi-header.templates": [
	{
		"language": "*",
		"template": [
			"File: <<filepath>>",
			"Project: <<projectpath>>",
			"Created Date: <<filecreated('dddd MMMM Do YYYY')>>",
			"Author: <<author>>",
			"-----",
			"Last Modified: <<filecreated('dddd MMMM Do YYYY h:mm:ss a')>>",
			"Modified By: the developer formerly known as <<author>> at <<<authoremail>>>",
			"-----",
			"Copyright (c) <<year>> <<company>>"
			"-----",
			"HISTORY:",
			"Date      \tBy\tComments",
			"----------\t---\t---------------------------------------------------------"
		],
		"changeLogCaption": "HISTORY:",
		"changeLogHeaderLineCount": 2,
		"changeLogEntryTemplate": [
			"",
			"<<dateformat('DD-MM-YYYY')>>\t<<initials>>\t"
		]
	},
]

would give output similar to the following:

/*
 * File: \Users\me\Development\psioniq\myProject\src\myPrecious.js
 * Project: \Users\me\Development\psioniq\myProject
 * Created Date: Saturday December 31 2016
 * Author: Arthur Bodkin, esq
 * -----
 * Last Modified: Tuesday January 03 2017 09:37:28 am
 * Modified By: the developer formerly known as Uncle Jack Bodkin at <uncle.jack@psioniq.net>
 * -----
 * Copyright (c) 2016 psioniq Global Enterprises, Inc
 * -----
 * HISTORY:
 * Date      	By	Comment
 * ----------	---	-----------------------------------------------------------
 *
 * 14-07-2018	JB	Added a rabbit
 *
 * 12-05-2018	AB	Fixed the type of bug that only a mother could love.
 */

Questions about Change Logs

Can this be configured to not have a caption line?

No. The caption line is how the extension works out where to add the log entries. You may have edited the header manually, so there is no easy way for the extension to map the raw template back to the edited header.

Can it be configured to automatically add a log entry?

No, because this would be extremely annoying if an entry was added every time the file was saved (which would then expect you to add a comment which would necessitate another save which would add another entry which would require you to add a comment which would...).

Can I have comments on a separate line?

Yes. Just provide your own psi-header.templates[].changeLogEntryTemplate which allows you to define a multi line template. You could add a log entry template something like:

"psi-header.templates": [
	{
		...other settings...
		"changeLogEntryTemplate": [
			"<<dateformat('YYYY-MM-DD')>>  (<<author>>)",
			""
		]
	},
]

Why do I have to manually add the comment?

The most likely cause is that your Visual Studio Brain Implant(TM) module is not correctly configured for your instance of VSCode. Try facing your computer and moving your head in a figure of eight pattern to establish a connection. If this fails, move your fingers frantically up and down near the keyboard.

OR

I am not very good at working out what your comment should contain.

What if I need longer comments?

No problem. Just add additional lines to your comment/entry. New log entries are always added to the top of the log, so it doesn't care if you have changed the layout of earlier entries.

An Example Custom Configuration

In the following example:

  • Javascript and Typescript files will both use the custom template and configuration where language = "javascript".
  • Lua will use it's own custom configuration (language="lua"), but will use the global custom template (language = "*").
  • All other languages will use the global custom template (language = "*") and the built in configuration settings because there is no custom global psi-header.lang-config defined.
  • changes tracking is turned on, but will skip Markdown, JSON and JSON with Comments files.
  • auto header creation is active, but will not save the file after inserting the heading.
{
	"psi-header.config": {
		"forceToTop": true,
		"blankLinesAfter": 6,
		"license": "Custom"
	},
	"psi-header.changes-tracking": {
		"isActive": true,
		"modAuthor": "Modified By: ",
		"modDate": "Last Modified: ",
		"modDateFormat": "date",
		"include": [],
		"exclude": [
			"markdown",
			"json",
			"jsonc"
		],
		"excludeGlob": [
			"out/**",
			"src/**/*.xyz"
		],
		"autoHeader": "manualSave"
	},
	"psi-header.license-text": [
		"All shall be well and all shall be well and all manner of things shall be well.",
		"Nope...we're doomed!"
	],
	"psi-header.variables": [
		["company", "psioniq"],
		["author", "Arthur Bodkin"],
		["authoremail", "art@psioniq.uk"]
	],
	"psi-header.lang-config": [
		{
			"language": "lua",
			"begin": "--[[",
			"prefix": "--",
			"end": "--]]",
			"blankLinesAfter": 0
		},
		{
			"language": "python",
			"begin": "###",
			"prefix": "# ",
			"end": "###",
			"blankLinesAfter": 0,
			"beforeHeader": [
				"#!/usr/bin/env python3",
				"# -*- coding:utf-8 -*-"
			]
		},
		{
			"language": "javascript",
			"begin": "/**",
			"prefix": " * ",
			"end": " */",
			"blankLinesAfter": 2,
			"forceToTop": false
		},
		{
			"language": "typescript",
			"mapTo": "javascript"
		}
	],
	"psi-header.templates": [
		{
			"language": "*",
			"template": [
				"File: <<filepath>>",
				"Project: <<projectpath>>",
				"Created Date: <<filecreated('dddd, MMMM Do YYYY, h:mm:ss a')>>",
				"Author: <<author>>",
				"-----",
				"Last Modified: ",
				"Modified By: ",
				"-----",
				"Copyright (c) <<year>> <<company>>"
				"",
				"<<licensetext>>",
				"-----"
				"HISTORY:",
				"Date      \tBy\tComments",
				"----------\t---\t----------------------------------------------------------"
			],
			"changeLogCaption": "HISTORY:",
			"changeLogHeaderLineCount": 2,
			"changeLogEntryTemplate": [
				"<<dateformat('YYYY-MM-DD')>>\t<<initials>>\t"
			]
		},
		{
			"language": "javascript",
			"template": [
				"File: <<filepath>>",
				"Project: <<projectpath>>",
				"Created Date: <<filecreated('dddd, MMMM Do YYYY, h:mm:ss a')>>",
				"Author: <<author>>",
				"-----",
				"Last Modified: ",
				"Modified By: ",
				"-----",
				"Copyright (c) <<year>> <<company>>",
				"------------------------------------",
				"Javascript will save your soul!"
			]
		},
		{
			"language": "typescript",
			"mapTo": "javascript"
		}
	]
}

Creating a Custom Template

This should be mostly obvious. The configuration of this extension separates out the syntactical language elements (comment Begin, comment End, etc) from the body of the template so that hopefully you will only need to create a single template.

For the variable placeholders, the variable names should be surrounded with << and >>. Do not add the prefix and suffix to your custom variable declarations! So in the template, the system variable filepath is written <<filepath>>. Easy, huh!

The default (built in) template is:

[
    "Filename: <<filepath>>",
    "Path: <<projectpath>>",
    "Created Date: <<filecreated('dddd, MMMM Do YYYY, h:mm:ss a')>>",
    "Author: <<author>>",
    "",
    "Copyright (c) <<year>> <<company>>"
]

...which, when rendered would produce the following (assuming JS):

/*
 * Filename: \Users\me\Development\psioniq\myProject\src\myPrecious.js
 * Path: \Users\me\Development\psioniq\myProject
 * Created Date: Saturday, December 31st 2016, 10:27:35 am
 * Author: David Quinn
 *
 * Copyright (c) 2016 psioniq Global Enterprises, Inc
 */

Known Issues

To report bugs, issues, suggestions use the github repository.

Cleaning up SPDX License Text

Refer to License Information for the extension's limitations on cleaning up SPDX license text.

Determining File Creation Time on Linux

The problem that led to this issue was "resolved" in v1.9.0 by providing a configuration option to determine what to do when the OS returns Epoch Zero as the file creation date.

The file creation routines may return an invalid value under some specific circumstances on Linux, and maddingly may depend on the filesystem where your VSCode project is stored (typically on an externally mounted NTFS partition). Usually it will show as a date time that may be on or around 1 January 1970. The problem relates to the filesystem's support of birthtime.

On investigation, it would appear that it is a problem with lack of support for birthtime in earlier versions of Linux. Support was added with the introduction of statx() in Linux Core in v4.11 in 2017, then to glibc in v2.28 in 2018. NodeJS accesses this via libuv and support for statx() was added to libuv v1.27.0 (Stable) in March 2019 as part of this PR. NodeJS supports this in v10.16.0 and v12.*.

Most annoyingly, affected Linux versions return a wrong date rather than nothing at all. So there is not a practical way to fix the creation date returned by the OS. However, in v1.9.0 we added a new configuration option creationDateZero that allows you to define what to do if the OS returns Epoch Zero. Refer to Global Options for details.

This link provides a good explanation of the problem.

Credits

This extension uses the following npm packages...thank you :):