Недавно я переместил в репозиторий множество MP3 файлов из разных мест. Я создавал новые имена файлов, используя теги ID3 (спасибо, TagLib-Sharp!), И я заметил, что получаю System.NotSupportedException:
"Данный формат пути не поддерживается."
Это было создано либо File.Copy(), либо Directory.CreateDirectory().
Не прошло много времени, чтобы понять, что мои имена файлов должны быть дезинфицированы. Поэтому я сделал очевидную вещь:
public static string SanitizePath_(string path, char replaceChar)
{
    string dir = Path.GetDirectoryName(path);
    foreach (char c in Path.GetInvalidPathChars())
        dir = dir.Replace(c, replaceChar);
    string name = Path.GetFileName(path);
    foreach (char c in Path.GetInvalidFileNameChars())
        name = name.Replace(c, replaceChar);
    return dir + name;
}
К моему удивлению, я продолжал получать исключения. Оказалось, что ':' не находится в наборе Path.GetInvalidPathChars(), потому что он действителен в корне пути. Полагаю, это имеет смысл, но это должно быть довольно распространенной проблемой. У кого-нибудь есть короткий код, который санирует путь? Самое тщательное, что я придумал это, но похоже, что это, вероятно, перебор.
    // replaces invalid characters with replaceChar
    public static string SanitizePath(string path, char replaceChar)
    {
        // construct a list of characters that can't show up in filenames.
        // need to do this because ":" is not in InvalidPathChars
        if (_BadChars == null)
        {
            _BadChars = new List<char>(Path.GetInvalidFileNameChars());
            _BadChars.AddRange(Path.GetInvalidPathChars());
            _BadChars = Utility.GetUnique<char>(_BadChars);
        }
        // remove root
        string root = Path.GetPathRoot(path);
        path = path.Remove(0, root.Length);
        // split on the directory separator character. Need to do this
        // because the separator is not valid in a filename.
        List<string> parts = new List<string>(path.Split(new char[]{Path.DirectorySeparatorChar}));
        // check each part to make sure it is valid.
        for (int i = 0; i < parts.Count; i++)
        {
            string part = parts[i];
            foreach (char c in _BadChars)
            {
                part = part.Replace(c, replaceChar);
            }
            parts[i] = part;
        }
        return root + Utility.Join(parts, Path.DirectorySeparatorChar.ToString());
    }
Любые улучшения, чтобы сделать эту функцию быстрее и менее барочной, было бы высоко оценено.
