如何用角度示意图覆盖文件?

Posted

技术标签:

【中文标题】如何用角度示意图覆盖文件?【英文标题】:How to overwrite file with angular schematics? 【发布时间】:2018-08-04 01:23:01 【问题描述】:

我想写一个Rule,它每次都会覆盖一个文件。在下面并将MergeStrategy 设置为Overwrite

collection.json


  "$schema": "../node_modules/@angular-devkit/schematics/collection-schema.json",
  "schematics": 
    "function": 
      "aliases": [ "fn" ],
      "factory": "./function",
      "description": "Create a function.",
      "schema": "./function/schema.json"
    
  

function/index.ts

export default function(options: FunctionOptions): Rule 
  options.path = options.path ? normalize(options.path) : options.path;
  const sourceDir = options.sourceDir;
  if (!sourceDir) 
    throw new SchematicsException(`sourceDir option is required.`);
  

  const templateSource: Source = apply(
    url('./files'),
    [
      template(
        ...strings,
        ...options,
      ),
      move(sourceDir),
    ]
  )

  return mergeWith(templateSource, MergeStrategy.Overwrite);


files/__path__/__name@dasherize__.ts

export function <%= camelize(name) %>(): void 

我运行schematics .:function --name=test --dry-run=false 我得到了

创建 /src/app/test.ts(33 字节)

但是,第二次。

错误! /src/app/test.ts 已经存在。

它不应该覆盖文件test.ts而不出错吗?

编辑:

所有答案都有效并且很棒,但似乎它们是变通方法,没有明显的“正确”答案,并且可能基于偏好/自以为是。所以不知道如何标记为已回答。

【问题讨论】:

实际上不,默认行为是保留文件原样。您可以使用--force 标志来强制覆盖文件,如下所示:schematics .:function --name=test --dry-run=false --force。 BTW:你在模板函数中为...strings 导入了什么? 我可能不明白MergeStrategy.Overwrite 的含义,如果那不是行为那么MergeStrategy.Overwrite 的意义/意义是什么。 哦,对不起,我完全错过了MergeStrategy.Overwrite,现在我同意它应该完全有效。至少在归档的 angular/dev-kit 项目中曾经存在过一个问题:github.com/angular/devkit/issues/745 github.com/angular/angular-cli/issues/11337 @jlang import strings from '@angular-devkit/core'; 【参考方案1】:

如果您要替换许多文件,这可能并不理想。我在向项目添加样板时遇到了替换 favicon.ico 的问题。我使用的解决方案是先显式删除它。

export function scaffold(options: any): Rule 
  return (tree: Tree, _context: SchematicContext) => 
  
    tree.delete('src/favicon.ico');

    const templateSource = apply(url('./files'), [
      template(
        ...options,
      ),
      move('.'),
    ]);

    return chain([
      branchAndMerge(chain([
        mergeWith(templateSource, MergeStrategy.Overwrite),
      ]), MergeStrategy.AllowOverwriteConflict),
    ])(tree, _context);
  ;

注意:此方法不再适用于当前版本的 Schematics。

以下似乎适用于 0.6.8 原理图。

export function scaffold(options: any): Rule 
  return (tree: Tree, _context: SchematicContext) => 
  
    const templateSource = apply(url('./files'), [
      template(
        ...options,
      ),
      move('.'),
    ]);

    return chain([
      branchAndMerge(chain([
        mergeWith(templateSource, MergeStrategy.Overwrite),
      ]), MergeStrategy.Overwrite),
    ])(tree, _context);
  ;

作为奖励,我不再需要明确删除文件。

【讨论】:

【参考方案2】:

我遇到了同样的问题。我偶然发现在apply 中添加forEach 可以删除文件并创建新文件。这是@angular-devkit/schematics-cli@0.6.8。

export function indexPage(options: any): Rule 
    return (tree: Tree, _context: SchematicContext) => 
        const rule = mergeWith(
            apply(url('./files'), [
                template( ...options ),
                forEach((fileEntry: FileEntry) => 
                    // Just by adding this is allows the file to be overwritten if it already exists
                    if (tree.exists(fileEntry.path)) return null;
                    return fileEntry;
                )

            ])
        );

        return rule(tree, _context);
    ;

【讨论】:

该死的,这不适用于嵌套文件夹结构。当心! 我将它与嵌套文件夹一起使用,深度为两层。但是我确实发现如果文件不存在会出错,所以我必须先添加一个检查是否存在。我会更新我的答案。 对于 forEach() 最后很重要,我发现 ie 在 move() 之后 这个答案不正确。它不会覆盖现有文件 - 只会忽略它们而不抛出“Schematics process failed”错误。 @cgatian 的回答确实会覆盖现有文件。【参考方案3】:

修改克里斯的回答我能够想出以下解决方案:

export function applyWithOverwrite(source: Source, rules: Rule[]): Rule 
  return (tree: Tree, _context: SchematicContext) => 
    const rule = mergeWith(
      apply(source, [
        ...rules,
        forEach((fileEntry) => 
          if (tree.exists(fileEntry.path)) 
            tree.overwrite(fileEntry.path, fileEntry.content);
            return null;
          
          return fileEntry;
        ),

      ]),
    );

    return rule(tree, _context);
  ;

将您对apply 的使用替换为对该函数的调用。

 applyWithOverwrite(url('./files/stylelint'), [
        template(
          dot: '.',
        ),
      ]),

【讨论】:

这个解决方案实际上覆盖了现有文件(而不是接受的文件),谢谢 :) N.B.由于某种原因,WebStorm 看不到它可以从哪里导入 forEach 和 FileEntry;它应该从 '@angular-devkit/schematics' 导入 如果forEach 之前的...rules 之一创建了该文件,则tree.exists(fileEntry.path) 将评估为true。换句话说,该文件可能实际上并不存在于您的文件系统中,而仅存在于由原理图构建的虚拟树中。考虑到您的用例,这可能很重要,也可能无关紧要。【参考方案4】:

感谢@cgatian 的提示! 不幸的是,它无法将模板 move 放到某个位置,所以我不得不添加:

forEach(fileEntry => 
  const destPath = join(options.output, fileEntry.path);
  if (tree.exists(destPath)) 
    tree.overwrite(destPath, fileEntry.content);
   else 
    tree.create(destPath, fileEntry.content);
  
  return null;
)

直到MergeStrategy 按预期工作, 这将在options.output 中传递目标路径!

再次感谢!

【讨论】:

【参考方案5】:

使用--force 参数: ng g c component-name --force

【讨论】:

以上是关于如何用角度示意图覆盖文件?的主要内容,如果未能解决你的问题,请参考以下文章

如何用 C# 覆盖二进制文件中的特定字节?

如何用新文件覆盖 log4j2.xml 中指定的日志文件名?

Objective C 类扩展 - 如何用 readwrite 方法覆盖只读?

如何用 0x00 覆盖所有可用磁盘空间? [关闭]

如何用自己的主题覆盖yii2.0中的当前主题

如何用批处理,写入hosts文件