blob: 93a8c2e31afa27634d90dcaf4beec750b97a6f1b [file] [log] [blame]
# Copyright 2022 The Pigweed Authors
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may not
# use this file except in compliance with the License. You may obtain a copy of
# the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations under
# the License.
"""Configure Visual Studio Code (VSC) for Pigweed projects.
VSC recognizes three sources of configurable settings:
1. Project settings, stored in {project root}/.vscode/settings.json
2. Workspace settings, stored in (workspace root)/.vscode/settings.json;
a workspace is a collection of projects/repositories that are worked on
together in a single VSC instance
3. The user's personal settings, which are stored somewhere in the user's home
directory, and are applied to all instances of VSC
This provides three levels of settings cascading:
Workspace <- Project <- User
... where the arrow indicates the ability to override.
Out of these three, only project settings are useful to Pigweed projects. User
settings are essentially global and outside the scope of Pigweed. Workspaces are
seldom used and don't work well with the Pigweed directory structure.
Nonetheless, we want a three-tiered settings structure for Pigweed projects too:
A. Default settings provided by Pigweed, configuring VSC to use IDE features
B. Project-level overrides that downstream projects may define
C. User-level overrides that individual users may define
We accomplish all of that with only the project settings described in #1 above.
Default settings are defined in this module. Project settings can be defined in
.vscode/pw_project_settings.json and should be checked into the repository. User
settings can be defined in .vscode/pw_user_settings.json and should not be
checked into the repository. None of these settings have any effect until they
are merged into VSC's settings (.vscode/settings.json) via the functions in this
module. Those resulting settings are system-specific and should also not be
checked into the repository.
We provide the same structure to both tasks and extensions as well. Defaults
are provided by Pigweed, can be augmented or overridden at the project level
with .vscode/pw_project_tasks.json and .vscode/pw_project_extensions.json,
can be augmented or overridden by an individual developer with
.vscode/pw_user_tasks.json and .vscode/pw_user.extensions.json, and none of
this takes effect until they are merged into VSC's active settings files
(.vscode/tasks.json and .vscode/extensions.json) by running the appropriate
command.
"""
# TODO(chadnorvell): Import collections.OrderedDict when we don't need to
# support Python 3.8 anymore.
from enum import Enum
import json
import os
from pathlib import Path
import platform
from typing import Any, Dict, List, OrderedDict
from pw_ide.activate import BashShellModifier
from pw_ide.cpp import ClangdSettings
from pw_ide.editors import (EditorSettingsDict, EditorSettingsManager,
EditorSettingsTypesWithDefaults, Json5FileFormat)
from pw_ide.python import PythonPaths
from pw_ide.settings import PigweedIdeSettings
def _vsc_os(system: str = platform.system()):
"""Return the OS tag that VSC expects."""
if system == 'Darwin':
return 'osx'
return system.lower()
def _activated_env() -> OrderedDict[str, Any]:
"""Return the environment diff needed to provide Pigweed activation.
The integrated terminal will already include the user's default environment
(e.g. from their shell init scripts). This provides the modifications to
the environment needed for Pigweed activation.
"""
# Not all environments have an actions.json, which this ultimately relies
# on (e.g. tests in CI). No problem, just return an empty dict instead.
try:
env = BashShellModifier(env_only=True,
path_var='${env:PATH}').modify_env().env_mod
except (FileNotFoundError, json.JSONDecodeError):
env = dict()
return OrderedDict(env)
def _local_terminal_integrated_env() -> Dict[str, Any]:
"""VSC setting to activate the integrated terminal."""
return {f'terminal.integrated.env.{_vsc_os()}': _activated_env()}
def _local_clangd_settings(ide_settings: PigweedIdeSettings) -> Dict[str, Any]:
"""VSC settings for running clangd with Pigweed paths."""
clangd_settings = ClangdSettings(ide_settings)
return {
'clangd.path': str(clangd_settings.clangd_path),
'clangd.arguments': clangd_settings.arguments,
}
def _local_python_settings() -> Dict[str, Any]:
"""VSC settings for finding the Python virtualenv."""
paths = PythonPaths()
return {
'python.defaultInterpreterPath': str(paths.interpreter),
'python.formatting.yapfPath': str(paths.bin_dir / 'yapf'),
}
# The order is preserved despite starting with a plain dict because in Python
# 3.6+, plain dicts are actually ordered as an implementation detail. This could
# break on interpreters other than CPython, or if the implementation changes in
# the future. However, for now, this is much more readable than the more robust
# alternative of instantiating with a list of tuples.
_DEFAULT_SETTINGS: EditorSettingsDict = OrderedDict({
"editor.detectIndentation": False,
"editor.rulers": [80],
"editor.tabSize": 2,
"files.associations": OrderedDict({"*.inc": "cpp"}),
"files.exclude": OrderedDict({
"**/*.egg-info": True,
"**/.mypy_cache": True,
"**/__pycache__": True,
".cache": True,
".cipd": True,
".environment": True,
".presubmit": True,
".pw_ide": True,
".pw_ide.user.yaml": True,
"bazel-bin": True,
"bazel-out": True,
"bazel-pigweed": True,
"bazel-testlogs": True,
"build": True,
"environment": True,
"node_modules": True,
"out": True}),
"files.insertFinalNewline": True,
"files.trimTrailingWhitespace": True,
"search.useGlobalIgnoreFiles": True,
"grunt.autoDetect": "off",
"gulp.autoDetect": "off",
"jake.autoDetect": "off",
"npm.autoDetect": "off",
"clangd.onConfigChanged": "restart",
"C_Cpp.intelliSenseEngine": "Disabled",
"[cpp]": OrderedDict(
{"editor.defaultFormatter": "llvm-vs-code-extensions.vscode-clangd"}),
"python.formatting.provider": "yapf",
"[python]": OrderedDict({"editor.tabSize": 4}),
"typescript.tsc.autoDetect": "off",
"[gn]": OrderedDict(
{"editor.defaultFormatter": "msedge-dev.gnls"}),
"[proto3]": OrderedDict(
{"editor.defaultFormatter": "zxh404.vscode-proto3"}),
"[restructuredtext]": OrderedDict({
"editor.quickSuggestions": OrderedDict({
"comments": "off",
"strings": "off",
"other": "off"})})
}) # yapf: disable
# pylint: disable=line-too-long
_DEFAULT_TASKS: EditorSettingsDict = OrderedDict({
"version": "2.0.0",
"tasks": [
{
"type": "shell",
"label": "Pigweed IDE: Format",
"command": "${workspaceFolder}/.pw_ide/python ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw format --fix'",
"problemMatcher": []
},
{
"type": "shell",
"label": "Pigweed IDE: Presubmit",
"command": "${workspaceFolder}/.pw_ide/python ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw presubmit'",
"problemMatcher": []
},
{
"label": "Pigweed IDE: Set Python Virtual Environment",
"command": "${command:python.setInterpreter}",
"problemMatcher": []
},
{
"label": "Pigweed IDE: Restart Python Language Server",
"command": "${command:python.analysis.restartLanguageServer}",
"problemMatcher": []
},
{
"label": "Pigweed IDE: Restart C++ Language Server",
"command": "${command:clangd.restart}",
"problemMatcher": []
},
{
"type": "shell",
"label": "Pigweed IDE: Process C++ Compilation Database from GN",
"command": "${workspaceFolder}/.pw_ide/python ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw ide cpp --gn --process out/compile_commands.json'",
"problemMatcher": []
},
{
"type": "shell",
"label": "Pigweed IDE: Setup",
"command": "python3 ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw ide setup'",
"problemMatcher": []
},
{
"type": "shell",
"label": "Pigweed IDE: Current C++ Code Analysis Target",
"command": "${workspaceFolder}/.pw_ide/python ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw ide cpp'",
"problemMatcher": []
},
{
"type": "shell",
"label": "Pigweed IDE: List C++ Code Analysis Targets",
"command": "${workspaceFolder}/.pw_ide/python ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw ide cpp --list'",
"problemMatcher": []
},
{
"type": "shell",
"label": "Pigweed IDE: Set C++ Code Analysis Target",
"command": "${workspaceFolder}/.pw_ide/python ${workspaceFolder}/pw_ide/py/pw_ide/activate.py -x 'pw ide cpp --set ${input:target}'",
"problemMatcher": []
}
],
"inputs": [
{
"id": "target",
"type": "promptString",
"description": "C++ code analysis target"
}
]
}) # yapf: disable
# pylint: enable=line-too-long
_DEFAULT_EXTENSIONS: EditorSettingsDict = OrderedDict({
"recommendations": [
"llvm-vs-code-extensions.vscode-clangd",
"ms-python.python",
"npclaudiu.vscode-gn",
"msedge-dev.gnls",
"zxh404.vscode-proto3",
"josetr.cmake-language-support-vscode"
],
"unwantedRecommendations": [
"ms-vscode.cpptools",
"persidskiy.vscode-gnformat"
]
}) # yapf: disable
def _default_settings(
pw_ide_settings: PigweedIdeSettings) -> EditorSettingsDict:
return OrderedDict({
**_DEFAULT_SETTINGS,
**_local_terminal_integrated_env(),
**_local_clangd_settings(pw_ide_settings),
**_local_python_settings(),
})
def _default_tasks(_pw_ide_settings: PigweedIdeSettings) -> EditorSettingsDict:
return _DEFAULT_TASKS
def _default_extensions(
_pw_ide_settings: PigweedIdeSettings) -> EditorSettingsDict:
return _DEFAULT_EXTENSIONS
DEFAULT_SETTINGS_PATH = Path(
os.path.expandvars('$PW_PROJECT_ROOT')) / '.vscode'
class VscSettingsType(Enum):
"""Visual Studio Code settings files.
VSC supports editor settings (``settings.json``), recommended
extensions (``extensions.json``), and tasks (``tasks.json``).
"""
SETTINGS = 'settings'
TASKS = 'tasks'
EXTENSIONS = 'extensions'
@classmethod
def all(cls) -> List['VscSettingsType']:
return list(cls)
class VscSettingsManager(EditorSettingsManager[VscSettingsType]):
"""Manages all settings for Visual Studio Code."""
default_settings_dir = DEFAULT_SETTINGS_PATH
file_format = Json5FileFormat()
types_with_defaults: EditorSettingsTypesWithDefaults = {
VscSettingsType.SETTINGS: _default_settings,
VscSettingsType.TASKS: _default_tasks,
VscSettingsType.EXTENSIONS: _default_extensions,
}