yannstatic/static/2019/07/11/debian-10-buster-une-distribution-qui-a-du-chien.html

1987 lines
203 KiB
HTML
Raw Normal View History

2024-10-31 20:18:37 +01:00
<!DOCTYPE html><html lang="fr">
<head><meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no"><title>Debian 10 Buster , une distribution qui a du chien - YannStatic</title>
<meta name="description" content="Debian 10 Buster">
<link rel="canonical" href="https://static.rnmkcy.eu/2019/07/11/debian-10-buster-une-distribution-qui-a-du-chien.html"><link rel="alternate" type="application/rss+xml" title="YannStatic" href="/feed.xml">
<!-- - include head/favicon.html - -->
<link rel="shortcut icon" type="image/png" href="/assets/favicon/favicon.png"><link rel="stylesheet" href="/assets/css/main.css"><link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.0.13/css/all.css" ><!-- start custom head snippets --><link rel="stylesheet" href="/assets/css/expand.css">
<!-- end custom head snippets --><script>(function() {
window.isArray = function(val) {
return Object.prototype.toString.call(val) === '[object Array]';
};
window.isString = function(val) {
return typeof val === 'string';
};
window.hasEvent = function(event) {
return 'on'.concat(event) in window.document;
};
window.isOverallScroller = function(node) {
return node === document.documentElement || node === document.body || node === window;
};
window.isFormElement = function(node) {
var tagName = node.tagName;
return tagName === 'INPUT' || tagName === 'SELECT' || tagName === 'TEXTAREA';
};
window.pageLoad = (function () {
var loaded = false, cbs = [];
window.addEventListener('load', function () {
var i;
loaded = true;
if (cbs.length > 0) {
for (i = 0; i < cbs.length; i++) {
cbs[i]();
}
}
});
return {
then: function(cb) {
cb && (loaded ? cb() : (cbs.push(cb)));
}
};
})();
})();
(function() {
window.throttle = function(func, wait) {
var args, result, thisArg, timeoutId, lastCalled = 0;
function trailingCall() {
lastCalled = new Date;
timeoutId = null;
result = func.apply(thisArg, args);
}
return function() {
var now = new Date,
remaining = wait - (now - lastCalled);
args = arguments;
thisArg = this;
if (remaining <= 0) {
clearTimeout(timeoutId);
timeoutId = null;
lastCalled = now;
result = func.apply(thisArg, args);
} else if (!timeoutId) {
timeoutId = setTimeout(trailingCall, remaining);
}
return result;
};
};
})();
(function() {
var Set = (function() {
var add = function(item) {
var i, data = this._data;
for (i = 0; i < data.length; i++) {
if (data[i] === item) {
return;
}
}
this.size ++;
data.push(item);
return data;
};
var Set = function(data) {
this.size = 0;
this._data = [];
var i;
if (data.length > 0) {
for (i = 0; i < data.length; i++) {
add.call(this, data[i]);
}
}
};
Set.prototype.add = add;
Set.prototype.get = function(index) { return this._data[index]; };
Set.prototype.has = function(item) {
var i, data = this._data;
for (i = 0; i < data.length; i++) {
if (this.get(i) === item) {
return true;
}
}
return false;
};
Set.prototype.is = function(map) {
if (map._data.length !== this._data.length) { return false; }
var i, j, flag, tData = this._data, mData = map._data;
for (i = 0; i < tData.length; i++) {
for (flag = false, j = 0; j < mData.length; j++) {
if (tData[i] === mData[j]) {
flag = true;
break;
}
}
if (!flag) { return false; }
}
return true;
};
Set.prototype.values = function() {
return this._data;
};
return Set;
})();
window.Lazyload = (function(doc) {
var queue = {js: [], css: []}, sources = {js: {}, css: {}}, context = this;
var createNode = function(name, attrs) {
var node = doc.createElement(name), attr;
for (attr in attrs) {
if (attrs.hasOwnProperty(attr)) {
node.setAttribute(attr, attrs[attr]);
}
}
return node;
};
var end = function(type, url) {
var s, q, qi, cbs, i, j, cur, val, flag;
if (type === 'js' || type ==='css') {
s = sources[type], q = queue[type];
s[url] = true;
for (i = 0; i < q.length; i++) {
cur = q[i];
if (cur.urls.has(url)) {
qi = cur, val = qi.urls.values();
qi && (cbs = qi.callbacks);
for (flag = true, j = 0; j < val.length; j++) {
cur = val[j];
if (!s[cur]) {
flag = false;
}
}
if (flag && cbs && cbs.length > 0) {
for (j = 0; j < cbs.length; j++) {
cbs[j].call(context);
}
qi.load = true;
}
}
}
}
};
var load = function(type, urls, callback) {
var s, q, qi, node, i, cur,
_urls = typeof urls === 'string' ? new Set([urls]) : new Set(urls), val, url;
if (type === 'js' || type ==='css') {
s = sources[type], q = queue[type];
for (i = 0; i < q.length; i++) {
cur = q[i];
if (_urls.is(cur.urls)) {
qi = cur;
break;
}
}
val = _urls.values();
if (qi) {
callback && (qi.load || qi.callbacks.push(callback));
callback && (qi.load && callback());
} else {
q.push({
urls: _urls,
callbacks: callback ? [callback] : [],
load: false
});
for (i = 0; i < val.length; i++) {
node = null, url = val[i];
if (s[url] === undefined) {
(type === 'js' ) && (node = createNode('script', { src: url }));
(type === 'css') && (node = createNode('link', { rel: 'stylesheet', href: url }));
if (node) {
node.onload = (function(type, url) {
return function() {
end(type, url);
};
})(type, url);
(doc.head || doc.body).appendChild(node);
s[url] = false;
}
}
}
}
}
};
return {
js: function(url, callback) {
load('js', url, callback);
},
css: function(url, callback) {
load('css', url, callback);
}
};
})(this.document);
})();
</script><script>
(function() {
var TEXT_VARIABLES = {
version: '2.2.6',
sources: {
font_awesome: 'https://use.fontawesome.com/releases/v5.0.13/css/all.css',
jquery: '/assets/js/jquery.min.js',
leancloud_js_sdk: '//cdn.jsdelivr.net/npm/leancloud-storage@3.13.2/dist/av-min.js',
chart: 'https://cdn.bootcss.com/Chart.js/2.7.2/Chart.bundle.min.js',
gitalk: {
js: 'https://cdn.bootcss.com/gitalk/1.2.2/gitalk.min.js',
css: 'https://cdn.bootcss.com/gitalk/1.2.2/gitalk.min.css'
},
valine: 'https://unpkg.com/valine/dist/Valine.min.js'
},
site: {
toc: {
selectors: 'h1,h2,h3'
}
},
paths: {
search_js: '/assets/search.js'
}
};
window.TEXT_VARIABLES = TEXT_VARIABLES;
})();
</script>
</head>
<body>
<div class="root" data-is-touch="false">
<div class="layout--page js-page-root"><!----><div class="page__main js-page-main page__viewport hide-footer has-aside has-aside cell cell--auto">
<div class="page__main-inner"><div class="page__header d-print-none"><header class="header"><div class="main">
<div class="header__title">
<div class="header__brand"><svg id="svg" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="400" height="478.9473684210526" viewBox="0, 0, 400,478.9473684210526"><g id="svgg"><path id="path0" d="M308.400 56.805 C 306.970 56.966,303.280 57.385,300.200 57.738 C 290.906 58.803,278.299 59.676,269.200 59.887 L 260.600 60.085 259.400 61.171 C 258.010 62.428,256.198 63.600,255.645 63.600 C 255.070 63.600,252.887 65.897,252.598 66.806 C 252.460 67.243,252.206 67.600,252.034 67.600 C 251.397 67.600,247.206 71.509,247.202 72.107 C 247.201 72.275,246.390 73.190,245.400 74.138 C 243.961 75.517,243.598 76.137,243.592 77.231 C 243.579 79.293,241.785 83.966,240.470 85.364 C 239.176 86.740,238.522 88.365,237.991 91.521 C 237.631 93.665,236.114 97.200,235.554 97.200 C 234.938 97.200,232.737 102.354,232.450 104.472 C 232.158 106.625,230.879 109.226,229.535 110.400 C 228.933 110.926,228.171 113.162,226.434 119.500 C 226.178 120.435,225.795 121.200,225.584 121.200 C 225.373 121.200,225.200 121.476,225.200 121.813 C 225.200 122.149,224.885 122.541,224.500 122.683 C 223.606 123.013,223.214 123.593,223.204 124.600 C 223.183 126.555,220.763 132.911,219.410 134.562 C 218.443 135.742,217.876 136.956,217.599 138.440 C 217.041 141.424,215.177 146.434,214.532 146.681 C 214.240 146.794,214.000 147.055,214.000 147.261 C 214.000 147.467,213.550 148.086,213.000 148.636 C 212.450 149.186,212.000 149.893,212.000 150.208 C 212.000 151.386,208.441 154.450,207.597 153.998 C 206.319 153.315,204.913 150.379,204.633 147.811 C 204.365 145.357,202.848 142.147,201.759 141.729 C 200.967 141.425,199.200 137.451,199.200 135.974 C 199.200 134.629,198.435 133.224,196.660 131.311 C 195.363 129.913,194.572 128.123,193.870 125.000 C 193.623 123.900,193.236 122.793,193.010 122.540 C 190.863 120.133,190.147 118.880,188.978 115.481 C 188.100 112.928,187.151 111.003,186.254 109.955 C 185.358 108.908,184.518 107.204,183.847 105.073 C 183.280 103.273,182.497 101.329,182.108 100.753 C 181.719 100.177,180.904 98.997,180.298 98.131 C 179.693 97.265,178.939 95.576,178.624 94.378 C 178.041 92.159,177.125 90.326,175.023 87.168 C 174.375 86.196,173.619 84.539,173.342 83.486 C 172.800 81.429,171.529 79.567,170.131 78.785 C 169.654 78.517,168.697 77.511,168.006 76.549 C 167.316 75.587,166.594 74.800,166.402 74.800 C 166.210 74.800,164.869 73.633,163.421 72.206 C 160.103 68.936,161.107 69.109,146.550 69.301 C 133.437 69.474,128.581 70.162,126.618 72.124 C 126.248 72.495,125.462 72.904,124.872 73.033 C 124.282 73.163,123.088 73.536,122.219 73.863 C 121.349 74.191,119.028 74.638,117.061 74.858 C 113.514 75.254,109.970 76.350,108.782 77.419 C 107.652 78.436,100.146 80.400,97.388 80.400 C 95.775 80.400,93.167 81.360,91.200 82.679 C 90.430 83.195,89.113 83.804,88.274 84.031 C 85.875 84.681,78.799 90.910,74.400 96.243 L 73.400 97.456 73.455 106.028 C 73.526 117.055,74.527 121.238,77.820 124.263 C 78.919 125.273,80.400 127.902,80.400 128.842 C 80.400 129.202,81.075 130.256,81.900 131.186 C 83.563 133.059,85.497 136.346,86.039 138.216 C 86.233 138.886,87.203 140.207,88.196 141.153 C 89.188 142.098,90.000 143.104,90.000 143.388 C 90.000 144.337,92.129 148.594,92.869 149.123 C 93.271 149.410,93.600 149.831,93.600 150.059 C 93.600 150.286,93.932 150.771,94.337 151.136 C 94.743 151.501,95.598 153.004,96.237 154.475 C 96.877 155.947,97.760 157.351,98.200 157.596 C 98.640 157.841,99.900 159.943,101.000 162.267 C 102.207 164.817,103.327 166.644,103.825 166.876 C 104.278 167.087,105.065 168.101,105.573 169.130 C 107.658 173.348,108.097 174.093,110.006 176.647 C 111.103 178.114,112.000 179.725,112.000 180.227 C 112.000 181.048,113.425 183.163,114.678 184.200 C 115.295 184.711,117.396 188.733,117.720 190.022 C 117.855 190.562,118.603 191.633,119.381 192.402 C 120.160 193.171,121.496 195.258,122.351 197.039 C 123.206 198.820,124.167 200.378,124.487 200.501 C 124.807 200.624,125.953 202.496,127.034 204.662 C 128.114 206.828,129.676 209.299,130.505 210.153 C 131.333 211.007,132.124 212.177,132.262 212.753 C 132.618 214.239,134.291 217.048,136.288 219.5
" href="/">YannStatic</a></div><!--<button class="button button--secondary button--circle search-button js-search-toggle"><i class="fas fa-search"></i></button>--><!-- <li><button class="button button--secondary button--circle search-button js-search-toggle"><i class="fas fa-search"></i></button></li> -->
<!-- Champ de recherche -->
<div id="searchbox" class="search search--dark" style="visibility: visible">
<div class="main">
<div class="search__header"></div>
<div class="search-bar">
<div class="search-box js-search-box">
<div class="search-box__icon-search"><i class="fas fa-search"></i></div>
<input id="search-input" type="text" />
<!-- <div class="search-box__icon-clear js-icon-clear">
<a><i class="fas fa-times"></i></a>
</div> -->
</div>
</div>
</div>
</div>
<!-- Script pointing to search-script.js -->
<script>/*!
* Simple-Jekyll-Search
* Copyright 2015-2020, Christian Fei
* Licensed under the MIT License.
*/
(function(){
'use strict'
var _$Templater_7 = {
compile: compile,
setOptions: setOptions
}
const options = {}
options.pattern = /\{(.*?)\}/g
options.template = ''
options.middleware = function () {}
function setOptions (_options) {
options.pattern = _options.pattern || options.pattern
options.template = _options.template || options.template
if (typeof _options.middleware === 'function') {
options.middleware = _options.middleware
}
}
function compile (data) {
return options.template.replace(options.pattern, function (match, prop) {
const value = options.middleware(prop, data[prop], options.template)
if (typeof value !== 'undefined') {
return value
}
return data[prop] || match
})
}
'use strict';
function fuzzysearch (needle, haystack) {
var tlen = haystack.length;
var qlen = needle.length;
if (qlen > tlen) {
return false;
}
if (qlen === tlen) {
return needle === haystack;
}
outer: for (var i = 0, j = 0; i < qlen; i++) {
var nch = needle.charCodeAt(i);
while (j < tlen) {
if (haystack.charCodeAt(j++) === nch) {
continue outer;
}
}
return false;
}
return true;
}
var _$fuzzysearch_1 = fuzzysearch;
'use strict'
/* removed: const _$fuzzysearch_1 = require('fuzzysearch') */;
var _$FuzzySearchStrategy_5 = new FuzzySearchStrategy()
function FuzzySearchStrategy () {
this.matches = function (string, crit) {
return _$fuzzysearch_1(crit.toLowerCase(), string.toLowerCase())
}
}
'use strict'
var _$LiteralSearchStrategy_6 = new LiteralSearchStrategy()
function LiteralSearchStrategy () {
this.matches = function (str, crit) {
if (!str) return false
str = str.trim().toLowerCase()
crit = crit.trim().toLowerCase()
return crit.split(' ').filter(function (word) {
return str.indexOf(word) >= 0
}).length === crit.split(' ').length
}
}
'use strict'
var _$Repository_4 = {
put: put,
clear: clear,
search: search,
setOptions: __setOptions_4
}
/* removed: const _$FuzzySearchStrategy_5 = require('./SearchStrategies/FuzzySearchStrategy') */;
/* removed: const _$LiteralSearchStrategy_6 = require('./SearchStrategies/LiteralSearchStrategy') */;
function NoSort () {
return 0
}
const data = []
let opt = {}
opt.fuzzy = false
opt.limit = 10
opt.searchStrategy = opt.fuzzy ? _$FuzzySearchStrategy_5 : _$LiteralSearchStrategy_6
opt.sort = NoSort
opt.exclude = []
function put (data) {
if (isObject(data)) {
return addObject(data)
}
if (isArray(data)) {
return addArray(data)
}
return undefined
}
function clear () {
data.length = 0
return data
}
function isObject (obj) {
return Boolean(obj) && Object.prototype.toString.call(obj) === '[object Object]'
}
function isArray (obj) {
return Boolean(obj) && Object.prototype.toString.call(obj) === '[object Array]'
}
function addObject (_data) {
data.push(_data)
return data
}
function addArray (_data) {
const added = []
clear()
for (let i = 0, len = _data.length; i < len; i++) {
if (isObject(_data[i])) {
added.push(addObject(_data[i]))
}
}
return added
}
function search (crit) {
if (!crit) {
return []
}
return findMatches(data, crit, opt.searchStrategy, opt).sort(opt.sort)
}
function __setOptions_4 (_opt) {
opt = _opt || {}
opt.fuzzy = _opt.fuzzy || false
opt.limit = _opt.limit || 10
opt.searchStrategy = _opt.fuzzy ? _$FuzzySearchStrategy_5 : _$LiteralSearchStrategy_6
opt.sort = _opt.sort || NoSort
opt.exclude = _opt.exclude || []
}
function findMatches (data, crit, strategy, opt) {
const matches = []
for (let i = 0; i < data.length && matches.length < opt.limit; i++) {
const match = findMatchesInObject(data[i], crit, strategy, opt)
if (match) {
matches.push(match)
}
}
return matches
}
function findMatchesInObject (obj, crit, strategy, opt) {
for (const key in obj) {
if (!isExcluded(obj[key], opt.exclude) && strategy.matches(obj[key], crit)) {
return obj
}
}
}
function isExcluded (term, excludedTerms) {
for (let i = 0, len = excludedTerms.length; i < len; i++) {
const excludedTerm = excludedTerms[i]
if (new RegExp(excludedTerm).test(term)) {
return true
}
}
return false
}
/* globals ActiveXObject:false */
'use strict'
var _$JSONLoader_2 = {
load: load
}
function load (location, callback) {
const xhr = getXHR()
xhr.open('GET', location, true)
xhr.onreadystatechange = createStateChangeListener(xhr, callback)
xhr.send()
}
function createStateChangeListener (xhr, callback) {
return function () {
if (xhr.readyState === 4 && xhr.status === 200) {
try {
callback(null, JSON.parse(xhr.responseText))
} catch (err) {
callback(err, null)
}
}
}
}
function getXHR () {
return window.XMLHttpRequest ? new window.XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP')
}
'use strict'
var _$OptionsValidator_3 = function OptionsValidator (params) {
if (!validateParams(params)) {
throw new Error('-- OptionsValidator: required options missing')
}
if (!(this instanceof OptionsValidator)) {
return new OptionsValidator(params)
}
const requiredOptions = params.required
this.getRequiredOptions = function () {
return requiredOptions
}
this.validate = function (parameters) {
const errors = []
requiredOptions.forEach(function (requiredOptionName) {
if (typeof parameters[requiredOptionName] === 'undefined') {
errors.push(requiredOptionName)
}
})
return errors
}
function validateParams (params) {
if (!params) {
return false
}
return typeof params.required !== 'undefined' && params.required instanceof Array
}
}
'use strict'
var _$utils_9 = {
merge: merge,
isJSON: isJSON
}
function merge (defaultParams, mergeParams) {
const mergedOptions = {}
for (const option in defaultParams) {
mergedOptions[option] = defaultParams[option]
if (typeof mergeParams[option] !== 'undefined') {
mergedOptions[option] = mergeParams[option]
}
}
return mergedOptions
}
function isJSON (json) {
try {
if (json instanceof Object && JSON.parse(JSON.stringify(json))) {
return true
}
return false
} catch (err) {
return false
}
}
var _$src_8 = {};
(function (window) {
'use strict'
let options = {
searchInput: null,
resultsContainer: null,
json: [],
success: Function.prototype,
searchResultTemplate: '<li><a href="{url}" title="{desc}">{title}</a></li>',
templateMiddleware: Function.prototype,
sortMiddleware: function () {
return 0
},
noResultsText: 'No results found',
limit: 10,
fuzzy: false,
debounceTime: null,
exclude: []
}
let debounceTimerHandle
const debounce = function (func, delayMillis) {
if (delayMillis) {
clearTimeout(debounceTimerHandle)
debounceTimerHandle = setTimeout(func, delayMillis)
} else {
func.call()
}
}
const requiredOptions = ['searchInput', 'resultsContainer', 'json']
/* removed: const _$Templater_7 = require('./Templater') */;
/* removed: const _$Repository_4 = require('./Repository') */;
/* removed: const _$JSONLoader_2 = require('./JSONLoader') */;
const optionsValidator = _$OptionsValidator_3({
required: requiredOptions
})
/* removed: const _$utils_9 = require('./utils') */;
window.SimpleJekyllSearch = function (_options) {
const errors = optionsValidator.validate(_options)
if (errors.length > 0) {
throwError('You must specify the following required options: ' + requiredOptions)
}
options = _$utils_9.merge(options, _options)
_$Templater_7.setOptions({
template: options.searchResultTemplate,
middleware: options.templateMiddleware
})
_$Repository_4.setOptions({
fuzzy: options.fuzzy,
limit: options.limit,
sort: options.sortMiddleware,
exclude: options.exclude
})
if (_$utils_9.isJSON(options.json)) {
initWithJSON(options.json)
} else {
initWithURL(options.json)
}
const rv = {
search: search
}
typeof options.success === 'function' && options.success.call(rv)
return rv
}
function initWithJSON (json) {
_$Repository_4.put(json)
registerInput()
}
function initWithURL (url) {
_$JSONLoader_2.load(url, function (err, json) {
if (err) {
throwError('failed to get JSON (' + url + ')')
}
initWithJSON(json)
})
}
function emptyResultsContainer () {
options.resultsContainer.innerHTML = ''
}
function appendToResultsContainer (text) {
options.resultsContainer.innerHTML += text
}
function registerInput () {
options.searchInput.addEventListener('input', function (e) {
if (isWhitelistedKey(e.which)) {
emptyResultsContainer()
debounce(function () { search(e.target.value) }, options.debounceTime)
}
})
}
function search (query) {
if (isValidQuery(query)) {
emptyResultsContainer()
render(_$Repository_4.search(query), query)
}
}
function render (results, query) {
const len = results.length
if (len === 0) {
return appendToResultsContainer(options.noResultsText)
}
for (let i = 0; i < len; i++) {
results[i].query = query
appendToResultsContainer(_$Templater_7.compile(results[i]))
}
}
function isValidQuery (query) {
return query && query.length > 0
}
function isWhitelistedKey (key) {
return [13, 16, 20, 37, 38, 39, 40, 91].indexOf(key) === -1
}
function throwError (message) {
throw new Error('SimpleJekyllSearch --- ' + message)
}
})(window)
}());
</script>
<!-- Configuration -->
<script>
SimpleJekyllSearch({
searchInput: document.getElementById('search-input'),
resultsContainer: document.getElementById('results-container'),
json: '/search.json',
//searchResultTemplate: '<li><a href="https://static.rnmkcy.eu{url}">{date}&nbsp;{title}</a></li>'
searchResultTemplate: '<li><a href="{url}">{date}&nbsp;{title}</a></li>'
})
</script>
<!-- Fin déclaration champ de recherche --></div><nav class="navigation">
2024-11-28 11:42:23 +01:00
<ul><li class="navigation__item"><a href="/archive.html">Etiquettes</a></li><li class="navigation__item"><a href="/htmldoc.html">Documents</a></li><li class="navigation__item"><a href="/liens_ttrss.html">Liens</a></li><li class="navigation__item"><a href="/syntaxe-markdown.html">Aide</a></li></ul>
2024-10-31 20:18:37 +01:00
</nav></div>
</header>
</div><div class="page__content"><div class ="main"><div class="grid grid--reverse">
<div class="col-main cell cell--auto"><!-- start custom main top snippet --><div id="results-container" class="search-result js-search-result"></div><!-- end custom main top snippet -->
<article itemscope itemtype="http://schema.org/Article"><div class="article__header"><header><h1 style="color:Tomato;">Debian 10 Buster , une distribution qui a du chien</h1></header></div><meta itemprop="headline" content="Debian 10 Buster , une distribution qui a du chien"><div class="article__info clearfix"><ul class="left-col menu"><li>
2024-11-08 14:10:33 +01:00
<a class="button button--secondary button--pill button--sm" style="color:#00FFFF" href="/archive.html?tag=debian">debian</a>
2024-10-31 20:18:37 +01:00
</li></ul><ul class="right-col menu"><li>
<i class="far fa-calendar-alt"></i>&nbsp;<span title="Création" style="color:#FF00FF">11&nbsp;juil.&nbsp;2019</span></li></ul></div><meta itemprop="datePublished" content="2019-07-11T00:00:00+02:00">
<meta itemprop="keywords" content="debian"><div class="js-article-content">
<div class="layout--article"><!-- start custom article top snippet -->
<style>
#myBtn {
display: none;
position: fixed;
bottom: 10px;
right: 10px;
z-index: 99;
font-size: 12px;
font-weight: bold;
border: none;
outline: none;
background-color: white;
color: black;
cursor: pointer;
padding: 5px;
border-radius: 4px;
}
#myBtn:hover {
background-color: #555;
}
</style>
<button onclick="topFunction()" id="myBtn" title="Haut de page">&#8679;</button>
<script>
//Get the button
var mybutton = document.getElementById("myBtn");
// When the user scrolls down 20px from the top of the document, show the button
window.onscroll = function() {scrollFunction()};
function scrollFunction() {
if (document.body.scrollTop > 20 || document.documentElement.scrollTop > 20) {
mybutton.style.display = "block";
} else {
mybutton.style.display = "none";
}
}
// When the user clicks on the button, scroll to the top of the document
function topFunction() {
document.body.scrollTop = 0;
document.documentElement.scrollTop = 0;
}
</script>
<!-- end custom article top snippet -->
<div class="article__content" itemprop="articleBody"><details>
<summary><b>Afficher/cacher Sommaire</b></summary>
<!-- affichage sommaire -->
<div class="toc-aside js-toc-root"></div>
</details><h1 id="debian-10-buster">Debian 10 Buster</h1>
<ul>
<li>URL: <a href="https://linuxfr.org/news/debian-10-buster-une-distribution-qui-a-du-chien">https://linuxfr.org/news/debian-10-buster-une-distribution-qui-a-du-chien</a></li>
<li>Authors: Collectif thomasv, antistress, j, M5oul, mzf, BAud, Davy Defaud, Arcaik, Xavier Claude, jihele, bolikahult, Arkem, jona, Raoul Volfoni, Apichat, Étienne BERSAC et cyberjunkie</li>
<li>Date: 2019-06-11T23:05:49+02:00</li>
<li>License: CC by-sa</li>
</ul>
<p>Debian GNU/Linux est une distribution communautaire entièrement construite avec des logiciels libres. Sa version 10, nom de code <em>Buster</em> (en référence au <a href="https://pixar.fandom.com/wiki/Buster">chien</a> dAndy dans <em><a href="https://fr.wikipedia.org/wiki/Toy_Story_2">Toy Story 2</a>)</em>, a été publiée le 6 juillet 2019.</p>
<p><img src="https://framapic.org/3VPBjhgYxkgc/Wb7DEAzZOIdC.png" alt="Debian 10" /></p>
<p><em>Buster</em> est disponible officiellement sur dix architectures différentes : AMD64, ARM64, ARMel, ARMhf, i386, MIPS, MIPS64el, MIPSel, PowerPC64el et s390x (<a href="https://lists.debian.org/debian-devel-announce/2019/04/msg00003.html">les mêmes</a> que pour <em><a href="https://linuxfr.org/news/debian-9-stretch-deploie-ses-tentacules">Stretch</a></em>, la précédente version).</p>
<p>Cette nouvelle version de Debian GNU/Linux contient plus de 51000 paquets, dont 15000 nouveaux. Par ailleurs, 6000 paquets ont été supprimés depuis <em>Stretch</em>.</p>
<p>Parmi les nouveautés, la sécurité est à lhonneur avec la prise en charge de <a href="https://fr.wikipedia.org/wiki/UEFI">SecureBoot</a> pour les architectures les plus répandues, lactivation d[[AppArmor]] sur les nouvelles installations, le choix de [[Wayland]] comme serveur daffichage par défaut pour [[GNOME]], ou encore les avancées concernant le chantier des compilations reproductibles.</p>
<hr />
<p><a href="https://www.debian.org/releases/buster/amd64/release-notes/ch-whats-new.fr.html">Nouveautés de Debian 10</a>
<a href="https://www.debian.org/releases/buster/amd64/release-notes/index.fr.html">Notes de publication pour Debian 10 (« Buster »)</a>
<a href="https://wiki.debian.org/DebianArt/Themes/futurePrototype">Debian Buster Artwork</a>
<a href="https://www.debian.org/doc/manuals/refcard/refcard">[PDF] Carte de référence Debian</a>
<a href="https://www.debian.org/News/2019/20190706">Annonce de la publication</a>
<a href="https://www.debian.org/News/2019/20190707">Annonce Debian Edu</a></p>
<hr />
<h1 id="présentation">Présentation</h1>
<p>Le projet [[Debian]] a été initié par <a href="https://fr.wikipedia.org/wiki/Ian_Murdock">Ian Murdock</a> en 1993. Cest lun des premiers systèmes dexploitations à utiliser un noyau Linux.</p>
<p>Le projet est développé internationalement par des bénévoles grâce à Internet. Le <a href="https://www.debian.org/devel/leader">Debian Project Leader</a> élu chaque année (actuellement <a href="https://linuxfr.org/news/sam-hartman-a-ete-elu-dpl-2019-debian-project-leader">Sam Hartman</a>, qui succède à Chris Lamb) guide cette communauté en sappuyant sur le <a href="https://www.debian.org/social_contract">Contrat social Debian</a> et <a href="https://www.debian.org/social_contract#guidelines">Les principes du logiciel libre selon Debian</a>.</p>
<p>Réputée pour sa stabilité, Debian sert de base à de nombreuses autres distributions. Le site <a href="https://fr.wikipedia.org/wiki/DistroWatch">DistroWatch</a> en dénombre <a href="https://distrowatch.com/search.php?ostype=All&amp;category=All&amp;origin=All&amp;basedon=Debian&amp;notbasedon=None&amp;desktop=All&amp;architecture=All&amp;status=Active">132 actives</a> dont les populaires <a href="https://ubuntu.com/">Ubuntu</a>, <a href="https://linuxmint.com/">Linux Mint</a>, <a href="https://tails.boum.org/index.fr.html">Tails</a> ou <a href="https://www.raspberrypi.org/downloads/raspbian/">Raspbian</a>.
Pour assurer cette stabilité, le cycle de développement dune nouvelle version est en général de plusieurs années. La sortie de <em>Buster</em> aura ainsi nécessité plus de deux ans de préparation.</p>
<p>Debian est une distribution généraliste. Elle fournit des outils autant pour la bureautique (LibreOffice, Thunderbird, GNOME, KDE, Xfce…), que pour le développement (GCC, Emacs, Vim, JDK, etc.), les serveurs Web (Apache, nginx), messagerie (Postfix…), virtualisation (KVM), conteneurisation (LXC, Docker…).</p>
<p>La liste des <a href="https://www.debian.org/users/">utilisateurs officiellement déclarés</a> contient des entreprises comme <a href="https://www.debian.org/users/com/backblaze">Blackblaze</a>, des organisations à but non lucratif comme <a href="https://www.debian.org/users/org/tuxfamily">TuxFamily</a>, des institutions éducatives ou encore des organisations gouvernementales comme l<a href="https://www.debian.org/users/gov/insee">INSEE</a>.</p>
<h1 id="cycle-de-développement">Cycle de développement</h1>
<p>À partir de la publication de <em>Stretch</em> en juin 2017, <em>Buster</em> est entrée dans sa phase de développement durant laquelle les paquets ajoutés dans le dépôt <em>unstable</em> migraient automatiquement dans <em>testing</em> au bout de quelques jours.</p>
<p>Après environ un an et demi de développement, les vannes se sont fermées et <em>Buster</em> est progressivement entrée dans sa <a href="https://release.debian.org/buster/freeze_policy.html">phase de gel</a>, composée de trois étapes.</p>
<h2 id="transition-freeze">Transition freeze</h2>
<p>Le gel de transition a débuté le 12 janvier 2019, interdisant les transitions de grande ampleur (comme les bibliothèques dont dépendent beaucoup de logiciels) et les migrations de paquets introduisant de nouvelles régressions.</p>
<h2 id="soft-freeze">Soft freeze</h2>
<p>Le gel léger a débuté le 12 février 2019, pendant lequel le délai de migration était fixé à au moins dix jours et interdisant lentrée (ou le retour) dans <em>testing</em> des paquets absents de <em>testing</em>.</p>
<h2 id="full-freeze">Full freeze</h2>
<p>Le gel complet a débuté le 12 mars 2019, qui a restreint les migrations aux corrections de bogues critiques pour la publication et aux bogues marqués importants dans les paquets optionnels.</p>
<p><img src="https://framapic.org/Hn2wMrTlvPa9/kMeJw7RIfZnO.image" alt="Debian 10" /></p>
<h1 id="nouveautés">Nouveautés</h1>
<h2 id="paquets-mis-àjour">Paquets mis à jour</h2>
<p>Côté plomberie :</p>
<ul>
<li>Linux 4.9 → <a href="https://www.developpez.com/actu/230249/Le-noyau-Linux-4-19-est-disponible-tour-d-horizon-des-nouveautes-qui-accompagnent-cette-version-LTS/">4.19</a>;</li>
<li>GCC 6.3 → <a href="https://linuxfr.org/news/sortie-de-gcc-8-1">8.3</a>;</li>
<li>glibc 2.24-11 → 2.28-10;</li>
<li>Python 3.5.3 → 3.7.3.</li>
</ul>
<p>Quelques environnements graphiques :</p>
<ul>
<li>GNOME 3.22 → <a href="https://linuxfr.org/news/parution-de-gnome-3-30">3.30.2</a> (qui inclut les changements <a href="https://linuxfr.org/news/gnome-fete-ses-20-ans">des</a> <a href="https://linuxfr.org/news/des-nouvelles-de-gnome-a-l-occasion-de-la-3-26">versions</a> <a href="https://linuxfr.org/news/gnome-3-28">intermédiaires</a>);</li>
<li>KDE Plasma 5.8 → <a href="https://kde.org/announcements/plasma-5.14.0.php">5.14</a>;</li>
<li>LXDE 9 → 10;</li>
<li>LXQt 0.11 → <a href="https://lxqt.org/release/2019/01/25/lxqt-0140/">0.14</a>;</li>
<li>MATE 1.16 → <a href="https://linuxfr.org/news/sortie-de-mate-1-20">1.20</a>;</li>
<li><a href="https://linuxfr.org/news/xfce-4-12-est-la">Xfce 4.12</a> (déjà dans <em>Stretch</em>);</li>
<li>Cinnamon 3.2 → <a href="https://blog.linuxmint.com/?p=3557">3.8</a>.</li>
</ul>
<p>Quelques applications :</p>
<ul>
<li>GIMP 2.8 → <a href="https://linuxfr.org/news/gimp-2-10-8-wilber-kid">2.10.8</a> (qui inclut les nouveautés <a href="https://linuxfr.org/news/gimp-2-10-6-rien-ne-nous-arrete">des</a> <a href="https://linuxfr.org/news/gimp-2-10-4-on-garde-le-rythme">versions</a> <a href="https://linuxfr.org/news/sortie-de-gimp-2-10-2">précédentes</a> de la série <a href="https://linuxfr.org/news/gimp-2-10-roule-au-gegl">2.10</a>);</li>
<li>Firefox ESR 52 puis 60 → <a href="https://linuxfr.org/news/firefox-60-et-60-esr">60</a> (Firefox <a href="https://developer.mozilla.org/fr/docs/Mozilla/Firefox/Firefox_ESR">ESR</a> bénéficie d<a href="https://lwn.net/Articles/676963/">une exception</a> pour recevoir les mises à jour de versions majeures dans la version stable de Debian);</li>
<li>LibreOffice 5.2 → <a href="https://wiki.documentfoundation.org/ReleaseNotes/6.1/fr">6.1</a> (qui inclut les nouveautés <a href="https://linuxfr.org/news/sortie-de-libreoffice-5-3">des</a> <a href="https://linuxfr.org/news/libreoffice-5-4-5">versions</a> <a href="https://linuxfr.org/news/sortie-de-libreoffice-6-0">intermédiaires</a>);</li>
<li>GnuPG 2.1 → 2.2;</li>
<li>PostgreSQL 9.6 → 11.3;</li>
<li>Flatpak 0.8 → <a href="https://github.com/flatpak/flatpak/releases/tag/1.2.0">1.2</a>.</li>
</ul>
<h2 id="nouveaux-paquets-intéressants">Nouveaux paquets intéressants</h2>
<ul>
<li><code class="language-plaintext highlighter-rouge">rustc</code>, le compilateur du langage <a href="https://fr.wikipedia.org/wiki/Rust_(langage)">Rust</a>, nécessaire notamment pour compiler les versions récentes du navigateur Firefox, est disponible en version 1.34 (en fait <a href="https://people.debian.org/~mafm/posts/2019/20190617_debian-gnulinux-riscv64-port-in-mid-2019/">plus de 500 paquets dans Debian dépendent déjà de Rust</a>);</li>
<li><code class="language-plaintext highlighter-rouge">matrix-synapse</code>, le serveur Matrix de référence, en version 0.99. La version 1, sortie récemment, <a href="https://matrix-team.pages.debian.net/blogue/2019/06/26/june-2019-matrix-on-debian-update/">arrivera bientôt dans les rétroportages</a>;</li>
<li><a href="https://linuxfr.org/users/mzf/journaux/un-tap-tempo-en-ligne-de-commande">taptempo</a>, <a href="https://linuxfr.org/tags/taptempo/public">quon ne présente plus</a>, en version 1.4.4;</li>
<li>[[DokuWiki]], absent de <em>Stretch</em> (voir le <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=854592">bogue n^(o) 854592</a>), est de retour dans <em>Buster</em>;</li>
<li><a href="https://the.exa.website/"><code class="language-plaintext highlighter-rouge">exa</code></a>, un remplaçant de <code class="language-plaintext highlighter-rouge">ls</code>;</li>
<li><a href="https://github.com/johang/vlc-bittorrent">vlc-bittorrent</a> fait <a href="http://people.skolelinux.org/pere/blog/Non_blocking_bittorrent_plugin_for_vlc.html">son entrée</a> dans Debian, il permet de lire les <a href="http://people.skolelinux.org/pere/blog/Web_browser_integration_of_VLC_with_Bittorrent_support.html">fichiers et liens magnets Torrent</a> avec VLC et depuis le navigateur… et <a href="http://people.skolelinux.org/pere/blog/Non_blocking_bittorrent_plugin_for_vlc.html">ça marche</a>!</li>
<li><a href="https://packages.debian.org/buster/hollywood">hollywood</a>, qui vous permettra de passer pour un hacker, un vrai;</li>
<li>[[LilyPond]], un programme de composition de partitions musicales, de retour après avoir été absent de <em>Stretch</em>; il est intéressant de constater que le <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=746005">bogue n^(o) 746005</a>, qui est à lorigine du retrait de LilyPond de <em>Stretch</em>, nest toujours pas résolu;</li>
<li><a href="https://linuxfr.org/news/movim-0-14-scotty">Movim</a>, en version 0.14;</li>
<li>[[poezio]], en version 0.12.1;</li>
<li><a href="https://github.com/speed47/spectre-meltdown-checker">spectre-meltdown-checker</a>.</li>
</ul>
<h2 id="paquets-supprimés-notables">Paquets supprimés notables</h2>
<ul>
<li><code class="language-plaintext highlighter-rouge">manpages-fr</code>, qui nétait plus maintenu depuis 2014, a été supprimé de <em>Buster</em> suite à louverture du <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=871564">bogue n^(o) 871564</a>; la traduction des pages de manuel a depuis repris, de même que leur empaquetage; hélas, le nouveau paquet na pas pu être prêt avant le gel;</li>
<li><a href="https://fr.wikipedia.org/wiki/Amarok_(logiciel)">Amarok</a>, suite à la transition de Qt4 vers Qt5 dans Debian (<a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=784448">Amarok est resté en Qt4</a>);</li>
<li><code class="language-plaintext highlighter-rouge">debian-doc-fr</code>, qui nétait plus mis à jour depuis… <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=897281">douze ans!</a></li>
<li>[[Redmine]], suite à la transition vers Ruby on Rails 5 dans Debian; contrairement au cas dAmarok, une version de Redmine fonctionnant avec RoR5 existe mais na <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=921049">pas été empaquetée à temps</a>.</li>
</ul>
<h2 id="iptables-est-remplacé-par-nftables">iptables est remplacé par nftables</h2>
<p>Le cadriciel <a href="https://www.debian.org/releases/buster/amd64/release-notes/ch-whats-new.fr.html#nftables">iptables est remplacé par nftables</a>. Debian <em>Buster</em> contiendra néanmoins <a href="https://wiki.debian.org/nftables">des outils pour pouvoir continuer à utiliser iptables</a>, mais recommande fortement de migrer vers nftables.</p>
<h2 id="apparmor-activé-par-défaut">AppArmor activé par défaut</h2>
<p>Sur les nouvelles installations, [[AppArmor]] sera dorénavant activé par défaut. AppArmor fournira par défaut des profils pour certains programmes (Apache, GnuPG, Bash…), tandis que dautres programmes seront livrés avec leur propre profil AppArmor. Enfin, le paquet <code class="language-plaintext highlighter-rouge">apparmor-profiles-extra</code> fournira des profils supplémentaires pour les paquets nembarquant pas leur propre profil de confinement AppArmor.</p>
<blockquote>
<p>Si vous faites la mise à niveau vers <em>Buster</em>, il suffit de vérifier que le paquet <code class="language-plaintext highlighter-rouge">apparmor</code> est bien installé pour profiter de ses fonctionnalités.</p>
</blockquote>
<h2 id="secureboot-avec-uefi">SecureBoot avec UEFI</h2>
<p>Absente lors de la sortie de <em>Stretch</em>, la prise en charge de SecureBoot avec [[UEFI]] est enfin dactualité avec <em>Buster</em>, pour les architectures AMD64, i386 et ARM64.</p>
<p><em>Cf.</em> <a href="https://debamax.com/blog/2019/04/19/an-overview-of-secure-boot-in-debian">https://debamax.com/blog/2019/04/19/an-overview-of-secure-boot-in-debian</a>.</p>
<blockquote>
<p>Pour profiter de SecureBoot à loccasion dune mise à niveau vers <em>Buster</em>, il faut installer les paquets <code class="language-plaintext highlighter-rouge">shim-signed</code>, <code class="language-plaintext highlighter-rouge">grub-efi-amd64-signed</code> ou <code class="language-plaintext highlighter-rouge">grub-efi-ia32-signed</code> et activer lUEFI.</p>
</blockquote>
<h2 id="usr-fusionné-pour-les-nouvelles-installations">/usr fusionné pour les nouvelles installations</h2>
<p>Sur les nouvelles installations, larborescence du système de fichiers est modifiée comme indiqué cidessous :</p>
<p><code class="language-plaintext highlighter-rouge">/bin</code><code class="language-plaintext highlighter-rouge">/usr/bin</code>;
<code class="language-plaintext highlighter-rouge">/sbin</code><code class="language-plaintext highlighter-rouge">/usr/sbin</code>;
<code class="language-plaintext highlighter-rouge">/lib</code><code class="language-plaintext highlighter-rouge">/usr/lib</code>.</p>
<p>Les anciens répertoires deviennent des liens symboliques pointant vers les nouveaux.</p>
<blockquote>
<p>En cas de mise à niveau vers <em>Buster</em>, le système de fichiers nest pas modifié, mais vous pouvez installer le paquet <code class="language-plaintext highlighter-rouge">usrmerge</code> pour lancer manuellement la conversion.</p>
</blockquote>
<h2 id="application-du-nouveau-schéma-de-nommage-pour-les-périphériques-réseau">Application du nouveau schéma de nommage pour les périphériques réseau</h2>
<p>Depuis <em>Stretch</em>, les nouvelles installations utilisent un nouveau schéma de nommage pour les périphériques réseau. Ainsi, les interfaces ne se nomment plus <code class="language-plaintext highlighter-rouge">eth0</code> ou <code class="language-plaintext highlighter-rouge">wlan0</code> mais ont plutôt des noms ressemblant à <code class="language-plaintext highlighter-rouge">enp1s1</code> ou <code class="language-plaintext highlighter-rouge">wlp3s0</code>.</p>
<p>À partir de <em>Buster</em>, ce changement sappliquera aussi aux installations existantes. <a href="https://www.debian.org/releases/buster/amd64/release-notes/ch-information.fr.html#migrate-interface-names">Une section des notes de publication</a> détaille comment anticiper ce changement ou, au contraire, conserver lancien schéma de nommage.</p>
<h2 id="adoption-de-wayland-dans-la-session-pardéfaut">Adoption de Wayland dans la session par défaut</h2>
<p>Avec la mise à jour de GNOME vers la version 3.30 est venu un changement important : le passage de X.Org à Wayland comme serveur daffichage par défaut pour cet environnement graphique. Si beaucoup de problèmes ont été <a href="http://libre-ouvert.toile-libre.org/?article224/test-de-gnome-wayland-sur-debian-pas-encore-tout-a-fait-ca">corrigés au fil du temps</a>, tout nest pas encore parfait, par exemple <a href="https://lists.debian.org/debian-accessibility/2019/02/msg00004.html">du point de vue de laccessibilité</a>. Ainsi, certains ont récemment fait connaître <a href="https://jmtd.net/log/buster_wayland/">leurs inquiétudes</a> concernant lampleur de cette transition et le peu de discussion qui la accompagnée.</p>
<p>Un <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=927667">rapport de bogue</a> a été écrit pour lancer cette discussion et il a été convenu quil restait trop peu de temps avant la publication de <em>Buster</em> pour revenir en arrière, et quon naurait pas assez de temps pour éprouver correctement un retour à X.Org. Par comparaison, <a href="https://fr.wikipedia.org/wiki/Fedora_(GNU/Linux)">Fedora</a> propose Wayland par défaut depuis la version 25 (sortie en novembre 2016, basée sur GNOME 3.22), mais aussi [[SUSE Linux Enterprise Desktop]] 15 (juin 2018, GNOME 3.26) et [[RHEL]] 8 (mai 2019, GNOME 3.28).</p>
<p>Toutes les applications ne tournent pas encore nativement sous Wayland (Firefox, Thunderbird et GIMP, par exemple, sont en cours de conversion), mais grâce au composant XWayland qui fournit une couche intermédiaire vous ny verrez que du feu.</p>
<p>Petite exception, toutefois, pour le gestionnaire de paquets graphique Synaptic, par exemple, dont le modèle de permission nest pas compatible avec les règles de sécurité de Wayland : il ne se lancera que dans un mode que lon pourrait qualifier de « lecture seule » :
<img src="https://pix.toile-libre.org/upload/original/1562626625.png" alt="Avertissement au lancement de Synaptic" />
À la place, il vous faudra utiliser un outil graphique comme la Logithèque GNOME, ou les outils habituels en ligne de commande.</p>
<h2 id="attention-au-coup-de-la-panne">Attention au coup de la panne</h2>
<p>Les notes de publication avertissent au sujet dun <a href="https://www.debian.org/releases/buster/amd64/release-notes/ch-information.fr.html#entropy-starvation">problème potentiel lors du démarrage</a> des systèmes sous <em>Buster</em>. En effet, faute davoir suffisamment dentropie, un système pourrait mettre jusquà plusieurs heures à démarrer. Un contournement par défaut est appliqué pour les systèmes AMD64 récents, et les notes de publication proposent dautres options pour les autres architectures et les machines virtuelles.</p>
<p>Un <a href="https://daniel-lange.com/archives/152-hello-buster.html">article passionnant</a> décrit ce problème en détail et énumère un grand nombre de solutions ainsi que leurs avantages et leurs inconvénients. Le <a href="https://wiki.debian.org/BoottimeEntropyStarvation">wiki de Debian</a> résume également ces informations.</p>
<h2 id="quel-navigateur-web-privilégier">Quel navigateur Web privilégier?</h2>
<p>Debian ne conseille pas véritablement <em>un</em> navigateur, mais prévient que tous les navigateurs proposés dans Debian ne sont pas égaux en termes de suivi des failles de sécurité. De ce point de vue, il vaut mieux <a href="https://www.debian.org/releases/buster/amd64/release-notes/ch-information.fr.html#browser-security">privilégier</a> un navigateur sappuyant sur le paquet <a href="https://webkitgtk.org/">webkit2gtk</a> (c.-à-d. <a href="https://packages.debian.org/buster/luakit">Luakit</a>, <a href="https://packages.debian.org/buster/midori">Midori</a>, <a href="https://packages.debian.org/buster/surf">surf</a> ou <a href="https://packages.debian.org/buster/epiphany-browser">GNOME Web</a>), Firefox ou encore Chromium.</p>
<h1 id="avancement-des-compilations-reproductibles">Avancement des compilations reproductibles</h1>
<p>Le projet <a href="https://reproducible-builds.org/">Reproducible Builds</a> a pour objectif de rendre la compilation des paquets déterministe afin de toujours obtenir le même binaire à partir des mêmes sources (et du même environnement de compilation). Cela permettra de vérifier que les paquets distribués sont bien ceux quils prétendent être.</p>
<p>Même si <em>Buster</em> serait en théorie reproductible à 93%, <a href="https://lists.debian.org/debian-devel/2019/03/msg00017.html">elle ne le sera en pratique quà 54%</a>. La différence sexplique par des paquets construits avant décembre 2016 qui nont jamais été reconstruits depuis, pour 24% du total, ainsi que 12% de paquets ayant reçu une <a href="https://wiki.debian.org/binNMU">modification binaire pour une architecture spécifique</a> et affectés par le <a href="https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=894441">bogue n^(o) 894441</a>.</p>
<p><img src="https://tests.reproducible-builds.org/debian/unstable/amd64/stats_pkg_state.png" alt="Évolution de reproductibilité depuis 2014" /></p>
<h1 id="un-point-sur-les-debian-pureblends">Un point sur les Debian Pure Blends</h1>
<p>Les <em>Debian Pure Blends</em> sont des variantes de Debian adaptées à certains groupes dutilisateurs. Il ne sagit pas de divergences (<em>forks</em>), mais bien densembles de paquets composés uniquement à partir de paquets Debian. Cela signifie que vous pouvez installer ces variantes soit en utilisant les médias (images dinstallation ou autonomes) fournis par chaque projet, soit en installant le métapaquet dédié sur une installation Debian classique.</p>
<p>Parmi les <a href="https://www.debian.org/blends/">nombreuses variantes</a>, on trouve notamment :</p>
<ul>
<li><a href="https://blends.debian.org/edu/">Debian Edu</a>, spécialisée dans le domaine de léducation et dont la version 10 est sortie <a href="https://www.debian.org/News/2019/20190707">un jour après <em>Buster</em></a>;</li>
<li><a href="https://blends.debian.org/astro/">Debian Astro</a>, pour les astronomes professionnels ou amateurs;</li>
<li><a href="https://blends.debian.org/accessibility">Debian Accessibility</a>, pour les personnes en situation de handicap;</li>
<li><a href="https://wiki.debian.org/FreedomBox">Freedombox</a>, pour les amateurs dautohébergement, dont la version 19.1 est sortie avec <em>Buster</em>.</li>
</ul>
<h1 id="les-titres-auxquels-vous-avez-échappé-pour-cettedépêche">Les titres auxquels vous avez échappé pour cette dépêche</h1>
<p><img src="https://framapic.org/jJ8Gdt8zRJ3A/L7uZVVoyBavH.jpg" alt="Teckel" /></p>
<ul>
<li>« Le chien entre en scène »;</li>
<li>« Une distribution qui ne manque pas de mordant »;</li>
<li>« Le meilleur ami du libriste »;</li>
<li>« La version qui décOUAF! » ;</li>
<li>« La distribution qui tonne »;</li>
<li>« <em>Buster</em>, nom dun OS »;</li>
<li>« <em>Buster</em> lâche son os »;</li>
<li>« <em>Buster</em>, une distrib au poil ».</li>
</ul>
<p>Nhésitez pas à proposer vos idées dans les commentaires!</p>
<h1 id="revue-de-presse-nonexhaustive">Revue de presse (non exhaustive)</h1>
<ul>
<li><a href="https://www.programmez.com/actualites/sortie-de-debian-linux-10-buster-29179">[<em>Programmez!</em>] Sortie de Debian Linux 10 « Buster »</a></li>
<li><a href="https://www.informaticien.be/index.ks?page=news_item&amp;id=27780">[<em>informaticien.be</em>] Debian 10 Buster : la nouvelle version stable disponible</a></li>
<li><a href="https://www.nextinpact.com/brief/debian-10-disponible---linux-4-19--secure-boot--apparmor-et-wayland-par-defaut-9323.htm">[<em>Next INpact</em>] Debian 10 disponible : Linux 4.19, Secure Boot, AppArmor et Wayland par défaut</a></li>
<li><a href="https://linux.developpez.com/actu/268842/Buster-la-version-10-de-Debian-Linux-est-disponible-avec-le-support-du-Secure-Boot-et-plus-de-paquets-que-Stretch/">[<em>Developpez.com</em>] Buster, la version 10 de Debian Linux est disponible avec le support du Secure Boot</a></li>
<li><a href="https://www.zdnet.com/article/debian-10-buster-linux-arrives/">[<em>ZDnet</em>] <em>Debian 10 Buster Linux arrives</em></a></li>
</ul>
<h1 id="et-après">Et après?</h1>
<p>Certains projets se dessinent déjà pour laprès <em>Buster</em> :</p>
<ul>
<li>le <a href="https://bits.debian.org/2019/06/100-papercuts-kickoff.html">projet « 100 papercuts »</a> sera lancé pendant <a href="https://debconf19.debconf.org/">DebConf19</a> et visera à corriger tous les petits problèmes qui gâchent lexpérience utilisateur;</li>
<li>la <a href="https://www.debian.org/releases/buster/amd64/release-notes/ch-information.en.html#deprecated-components">fin de la prise en charge de Python2</a>;</li>
<li>la <a href="https://lists.debian.org/debian-devel-announce/2019/07/msg00002.html">fin des téléversements binaires</a>; dorénavant, seuls les téléversements de paquets source seront autorisés.</li>
</ul>
<p>La version 11 de Debian sappellera <em><a href="https://lists.debian.org/debian-devel-announce/2016/07/msg00002.html">Bullseye</a></em> (le <a href="https://pixar.fandom.com/wiki/Bullseye">cheval</a> de <a href="https://linuxfr.org/news/sortie-de-la-debian-gnulinux-30-woody">Woody</a>) et sortira… quand elle sera prête!</p>
<p>Bon, cest pas tout ça, mais je fais quoi maintenant moi aujourdhui : jinstalle la nouvelle version de Debian ou je vais voir au cinéma le <a href="https://fr.wikipedia.org/wiki/Toy_Story_4">nouvel épisode de <em>Toy Story</em></a>?</p>
</div>
<div class="d-print-none"><footer class="article__footer"><meta itemprop="dateModified" content="2019-07-11T00:00:00+02:00"><!-- start custom article footer snippet -->
<!-- end custom article footer snippet -->
<!--
<div align="right"><a type="application/rss+xml" href="/feed.xml" title="S'abonner"><i class="fa fa-rss fa-2x"></i></a>
&emsp;</div>
-->
</footer>
<div class="article__section-navigator clearfix"><div class="previous"><span>PRÉCÉDENT</span><a href="/2019/07/10/OVH-vps626865-SSD1-KVM-OpenStack(yanspm.com).html">OVH VPS SSD 1 KVM OpenStack (1 vCore/2GoRam/20GoSSD) domaine yanspm.com</a></div><div class="next"><span>SUIVANT</span><a href="/2019/07/17/Archlinux-XFCE-PC2.html">Archlinux XFCE (ASRock QC5000M Quad-Core APU,GPT partition)</a></div></div></div>
</div>
<script>(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
$(function() {
var $this ,$scroll;
var $articleContent = $('.js-article-content');
var hasSidebar = $('.js-page-root').hasClass('layout--page--sidebar');
var scroll = hasSidebar ? '.js-page-main' : 'html, body';
$scroll = $(scroll);
$articleContent.find('.highlight').each(function() {
$this = $(this);
$this.attr('data-lang', $this.find('code').attr('data-lang'));
});
$articleContent.find('h1[id], h2[id], h3[id], h4[id], h5[id], h6[id]').each(function() {
$this = $(this);
$this.append($('<a class="anchor d-print-none" aria-hidden="true"></a>').html('<i class="fas fa-anchor"></i>'));
});
$articleContent.on('click', '.anchor', function() {
$scroll.scrollToAnchor('#' + $(this).parent().attr('id'), 400);
});
});
});
})();
</script>
</div><section class="page__comments d-print-none"></section></article><!-- start custom main bottom snippet -->
<!-- end custom main bottom snippet -->
</div>
</div></div></div></div>
</div><script>(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
var $body = $('body'), $window = $(window);
var $pageRoot = $('.js-page-root'), $pageMain = $('.js-page-main');
var activeCount = 0;
function modal(options) {
var $root = this, visible, onChange, hideWhenWindowScroll = false;
var scrollTop;
function setOptions(options) {
var _options = options || {};
visible = _options.initialVisible === undefined ? false : show;
onChange = _options.onChange;
hideWhenWindowScroll = _options.hideWhenWindowScroll;
}
function init() {
setState(visible);
}
function setState(isShow) {
if (isShow === visible) {
return;
}
visible = isShow;
if (visible) {
activeCount++;
scrollTop = $(window).scrollTop() || $pageMain.scrollTop();
$root.addClass('modal--show');
$pageMain.scrollTop(scrollTop);
activeCount === 1 && ($pageRoot.addClass('show-modal'), $body.addClass('of-hidden'));
hideWhenWindowScroll && window.hasEvent('touchstart') && $window.on('scroll', hide);
$window.on('keyup', handleKeyup);
} else {
activeCount > 0 && activeCount--;
$root.removeClass('modal--show');
$window.scrollTop(scrollTop);
activeCount === 0 && ($pageRoot.removeClass('show-modal'), $body.removeClass('of-hidden'));
hideWhenWindowScroll && window.hasEvent('touchstart') && $window.off('scroll', hide);
$window.off('keyup', handleKeyup);
}
onChange && onChange(visible);
}
function show() {
setState(true);
}
function hide() {
setState(false);
}
function handleKeyup(e) {
// Char Code: 27 ESC
if (e.which === 27) {
hide();
}
}
setOptions(options);
init();
return {
show: show,
hide: hide,
$el: $root
};
}
$.fn.modal = modal;
});
})();
</script><div class="modal modal--overflow page__search-modal d-print-none js-page-search-modal"><script>
(function () {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
// search panel
var search = (window.search || (window.search = {}));
var useDefaultSearchBox = window.useDefaultSearchBox === undefined ?
true : window.useDefaultSearchBox ;
var $searchModal = $('.js-page-search-modal');
var $searchToggle = $('.js-search-toggle');
var searchModal = $searchModal.modal({ onChange: handleModalChange, hideWhenWindowScroll: true });
var modalVisible = false;
search.searchModal = searchModal;
var $searchBox = null;
var $searchInput = null;
var $searchClear = null;
function getModalVisible() {
return modalVisible;
}
search.getModalVisible = getModalVisible;
function handleModalChange(visible) {
modalVisible = visible;
if (visible) {
search.onShow && search.onShow();
useDefaultSearchBox && $searchInput[0] && $searchInput[0].focus();
} else {
search.onShow && search.onHide();
useDefaultSearchBox && $searchInput[0] && $searchInput[0].blur();
setTimeout(function() {
useDefaultSearchBox && ($searchInput.val(''), $searchBox.removeClass('not-empty'));
search.clear && search.clear();
window.pageAsideAffix && window.pageAsideAffix.refresh();
}, 400);
}
}
$searchToggle.on('click', function() {
modalVisible ? searchModal.hide() : searchModal.show();
});
// Char Code: 83 S, 191 /
$(window).on('keyup', function(e) {
if (!modalVisible && !window.isFormElement(e.target || e.srcElement) && (e.which === 83 || e.which === 191)) {
modalVisible || searchModal.show();
}
});
if (useDefaultSearchBox) {
$searchBox = $('.js-search-box');
$searchInput = $searchBox.children('input');
$searchClear = $searchBox.children('.js-icon-clear');
search.getSearchInput = function() {
return $searchInput.get(0);
};
search.getVal = function() {
return $searchInput.val();
};
search.setVal = function(val) {
$searchInput.val(val);
};
$searchInput.on('focus', function() {
$(this).addClass('focus');
});
$searchInput.on('blur', function() {
$(this).removeClass('focus');
});
$searchInput.on('input', window.throttle(function() {
var val = $(this).val();
if (val === '' || typeof val !== 'string') {
search.clear && search.clear();
} else {
$searchBox.addClass('not-empty');
search.onInputNotEmpty && search.onInputNotEmpty(val);
}
}, 400));
$searchClear.on('click', function() {
$searchInput.val(''); $searchBox.removeClass('not-empty');
search.clear && search.clear();
});
}
});
})();
</script><div class="search search--dark">
<div class="main">
<div class="search__header">Recherche</div>
<div class="search-bar">
<div class="search-box js-search-box">
<div class="search-box__icon-search"><i class="fas fa-search"></i></div>
<input id="search-input" type="text" />
<div class="search-box__icon-clear js-icon-clear">
<a><i class="fas fa-times"></i></a>
</div>
</div>
<button class="button button--theme-dark button--pill search__cancel js-search-toggle">
Annuler</button>
</div>
<div id="results-container" class="search-result js-search-result"></div>
</div>
</div>
<!-- Script pointing to search-script.js -->
<script>/*!
* Simple-Jekyll-Search
* Copyright 2015-2020, Christian Fei
* Licensed under the MIT License.
*/
(function(){
'use strict'
var _$Templater_7 = {
compile: compile,
setOptions: setOptions
}
const options = {}
options.pattern = /\{(.*?)\}/g
options.template = ''
options.middleware = function () {}
function setOptions (_options) {
options.pattern = _options.pattern || options.pattern
options.template = _options.template || options.template
if (typeof _options.middleware === 'function') {
options.middleware = _options.middleware
}
}
function compile (data) {
return options.template.replace(options.pattern, function (match, prop) {
const value = options.middleware(prop, data[prop], options.template)
if (typeof value !== 'undefined') {
return value
}
return data[prop] || match
})
}
'use strict';
function fuzzysearch (needle, haystack) {
var tlen = haystack.length;
var qlen = needle.length;
if (qlen > tlen) {
return false;
}
if (qlen === tlen) {
return needle === haystack;
}
outer: for (var i = 0, j = 0; i < qlen; i++) {
var nch = needle.charCodeAt(i);
while (j < tlen) {
if (haystack.charCodeAt(j++) === nch) {
continue outer;
}
}
return false;
}
return true;
}
var _$fuzzysearch_1 = fuzzysearch;
'use strict'
/* removed: const _$fuzzysearch_1 = require('fuzzysearch') */;
var _$FuzzySearchStrategy_5 = new FuzzySearchStrategy()
function FuzzySearchStrategy () {
this.matches = function (string, crit) {
return _$fuzzysearch_1(crit.toLowerCase(), string.toLowerCase())
}
}
'use strict'
var _$LiteralSearchStrategy_6 = new LiteralSearchStrategy()
function LiteralSearchStrategy () {
this.matches = function (str, crit) {
if (!str) return false
str = str.trim().toLowerCase()
crit = crit.trim().toLowerCase()
return crit.split(' ').filter(function (word) {
return str.indexOf(word) >= 0
}).length === crit.split(' ').length
}
}
'use strict'
var _$Repository_4 = {
put: put,
clear: clear,
search: search,
setOptions: __setOptions_4
}
/* removed: const _$FuzzySearchStrategy_5 = require('./SearchStrategies/FuzzySearchStrategy') */;
/* removed: const _$LiteralSearchStrategy_6 = require('./SearchStrategies/LiteralSearchStrategy') */;
function NoSort () {
return 0
}
const data = []
let opt = {}
opt.fuzzy = false
opt.limit = 10
opt.searchStrategy = opt.fuzzy ? _$FuzzySearchStrategy_5 : _$LiteralSearchStrategy_6
opt.sort = NoSort
opt.exclude = []
function put (data) {
if (isObject(data)) {
return addObject(data)
}
if (isArray(data)) {
return addArray(data)
}
return undefined
}
function clear () {
data.length = 0
return data
}
function isObject (obj) {
return Boolean(obj) && Object.prototype.toString.call(obj) === '[object Object]'
}
function isArray (obj) {
return Boolean(obj) && Object.prototype.toString.call(obj) === '[object Array]'
}
function addObject (_data) {
data.push(_data)
return data
}
function addArray (_data) {
const added = []
clear()
for (let i = 0, len = _data.length; i < len; i++) {
if (isObject(_data[i])) {
added.push(addObject(_data[i]))
}
}
return added
}
function search (crit) {
if (!crit) {
return []
}
return findMatches(data, crit, opt.searchStrategy, opt).sort(opt.sort)
}
function __setOptions_4 (_opt) {
opt = _opt || {}
opt.fuzzy = _opt.fuzzy || false
opt.limit = _opt.limit || 10
opt.searchStrategy = _opt.fuzzy ? _$FuzzySearchStrategy_5 : _$LiteralSearchStrategy_6
opt.sort = _opt.sort || NoSort
opt.exclude = _opt.exclude || []
}
function findMatches (data, crit, strategy, opt) {
const matches = []
for (let i = 0; i < data.length && matches.length < opt.limit; i++) {
const match = findMatchesInObject(data[i], crit, strategy, opt)
if (match) {
matches.push(match)
}
}
return matches
}
function findMatchesInObject (obj, crit, strategy, opt) {
for (const key in obj) {
if (!isExcluded(obj[key], opt.exclude) && strategy.matches(obj[key], crit)) {
return obj
}
}
}
function isExcluded (term, excludedTerms) {
for (let i = 0, len = excludedTerms.length; i < len; i++) {
const excludedTerm = excludedTerms[i]
if (new RegExp(excludedTerm).test(term)) {
return true
}
}
return false
}
/* globals ActiveXObject:false */
'use strict'
var _$JSONLoader_2 = {
load: load
}
function load (location, callback) {
const xhr = getXHR()
xhr.open('GET', location, true)
xhr.onreadystatechange = createStateChangeListener(xhr, callback)
xhr.send()
}
function createStateChangeListener (xhr, callback) {
return function () {
if (xhr.readyState === 4 && xhr.status === 200) {
try {
callback(null, JSON.parse(xhr.responseText))
} catch (err) {
callback(err, null)
}
}
}
}
function getXHR () {
return window.XMLHttpRequest ? new window.XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP')
}
'use strict'
var _$OptionsValidator_3 = function OptionsValidator (params) {
if (!validateParams(params)) {
throw new Error('-- OptionsValidator: required options missing')
}
if (!(this instanceof OptionsValidator)) {
return new OptionsValidator(params)
}
const requiredOptions = params.required
this.getRequiredOptions = function () {
return requiredOptions
}
this.validate = function (parameters) {
const errors = []
requiredOptions.forEach(function (requiredOptionName) {
if (typeof parameters[requiredOptionName] === 'undefined') {
errors.push(requiredOptionName)
}
})
return errors
}
function validateParams (params) {
if (!params) {
return false
}
return typeof params.required !== 'undefined' && params.required instanceof Array
}
}
'use strict'
var _$utils_9 = {
merge: merge,
isJSON: isJSON
}
function merge (defaultParams, mergeParams) {
const mergedOptions = {}
for (const option in defaultParams) {
mergedOptions[option] = defaultParams[option]
if (typeof mergeParams[option] !== 'undefined') {
mergedOptions[option] = mergeParams[option]
}
}
return mergedOptions
}
function isJSON (json) {
try {
if (json instanceof Object && JSON.parse(JSON.stringify(json))) {
return true
}
return false
} catch (err) {
return false
}
}
var _$src_8 = {};
(function (window) {
'use strict'
let options = {
searchInput: null,
resultsContainer: null,
json: [],
success: Function.prototype,
searchResultTemplate: '<li><a href="{url}" title="{desc}">{title}</a></li>',
templateMiddleware: Function.prototype,
sortMiddleware: function () {
return 0
},
noResultsText: 'No results found',
limit: 10,
fuzzy: false,
debounceTime: null,
exclude: []
}
let debounceTimerHandle
const debounce = function (func, delayMillis) {
if (delayMillis) {
clearTimeout(debounceTimerHandle)
debounceTimerHandle = setTimeout(func, delayMillis)
} else {
func.call()
}
}
const requiredOptions = ['searchInput', 'resultsContainer', 'json']
/* removed: const _$Templater_7 = require('./Templater') */;
/* removed: const _$Repository_4 = require('./Repository') */;
/* removed: const _$JSONLoader_2 = require('./JSONLoader') */;
const optionsValidator = _$OptionsValidator_3({
required: requiredOptions
})
/* removed: const _$utils_9 = require('./utils') */;
window.SimpleJekyllSearch = function (_options) {
const errors = optionsValidator.validate(_options)
if (errors.length > 0) {
throwError('You must specify the following required options: ' + requiredOptions)
}
options = _$utils_9.merge(options, _options)
_$Templater_7.setOptions({
template: options.searchResultTemplate,
middleware: options.templateMiddleware
})
_$Repository_4.setOptions({
fuzzy: options.fuzzy,
limit: options.limit,
sort: options.sortMiddleware,
exclude: options.exclude
})
if (_$utils_9.isJSON(options.json)) {
initWithJSON(options.json)
} else {
initWithURL(options.json)
}
const rv = {
search: search
}
typeof options.success === 'function' && options.success.call(rv)
return rv
}
function initWithJSON (json) {
_$Repository_4.put(json)
registerInput()
}
function initWithURL (url) {
_$JSONLoader_2.load(url, function (err, json) {
if (err) {
throwError('failed to get JSON (' + url + ')')
}
initWithJSON(json)
})
}
function emptyResultsContainer () {
options.resultsContainer.innerHTML = ''
}
function appendToResultsContainer (text) {
options.resultsContainer.innerHTML += text
}
function registerInput () {
options.searchInput.addEventListener('input', function (e) {
if (isWhitelistedKey(e.which)) {
emptyResultsContainer()
debounce(function () { search(e.target.value) }, options.debounceTime)
}
})
}
function search (query) {
if (isValidQuery(query)) {
emptyResultsContainer()
render(_$Repository_4.search(query), query)
}
}
function render (results, query) {
const len = results.length
if (len === 0) {
return appendToResultsContainer(options.noResultsText)
}
for (let i = 0; i < len; i++) {
results[i].query = query
appendToResultsContainer(_$Templater_7.compile(results[i]))
}
}
function isValidQuery (query) {
return query && query.length > 0
}
function isWhitelistedKey (key) {
return [13, 16, 20, 37, 38, 39, 40, 91].indexOf(key) === -1
}
function throwError (message) {
throw new Error('SimpleJekyllSearch --- ' + message)
}
})(window)
}());
</script>
<!-- Configuration -->
<script>
SimpleJekyllSearch({
searchInput: document.getElementById('search-input'),
resultsContainer: document.getElementById('results-container'),
noResultsText: '<p>Aucun résultat!</p>',
json: '/search.json',
searchResultTemplate: '<li><a href="{url}">{date}&nbsp;{title}</a>&nbsp;(Création {create})</li>'
})
</script>
</div></div>
<script>(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
function scrollToAnchor(anchor, duration, callback) {
var $root = this;
$root.animate({ scrollTop: $(anchor).position().top }, duration, function() {
window.history.replaceState(null, '', window.location.href.split('#')[0] + anchor);
callback && callback();
});
}
$.fn.scrollToAnchor = scrollToAnchor;
});
})();
(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
function affix(options) {
var $root = this, $window = $(window), $scrollTarget, $scroll,
offsetBottom = 0, scrollTarget = window, scroll = window.document, disabled = false, isOverallScroller = true,
rootTop, rootLeft, rootHeight, scrollBottom, rootBottomTop,
hasInit = false, curState;
function setOptions(options) {
var _options = options || {};
_options.offsetBottom && (offsetBottom = _options.offsetBottom);
_options.scrollTarget && (scrollTarget = _options.scrollTarget);
_options.scroll && (scroll = _options.scroll);
_options.disabled !== undefined && (disabled = _options.disabled);
$scrollTarget = $(scrollTarget);
isOverallScroller = window.isOverallScroller($scrollTarget[0]);
$scroll = $(scroll);
}
function preCalc() {
top();
rootHeight = $root.outerHeight();
rootTop = $root.offset().top + (isOverallScroller ? 0 : $scrollTarget.scrollTop());
rootLeft = $root.offset().left;
}
function calc(needPreCalc) {
needPreCalc && preCalc();
scrollBottom = $scroll.outerHeight() - offsetBottom - rootHeight;
rootBottomTop = scrollBottom - rootTop;
}
function top() {
if (curState !== 'top') {
$root.removeClass('fixed').css({
left: 0,
top: 0
});
curState = 'top';
}
}
function fixed() {
if (curState !== 'fixed') {
$root.addClass('fixed').css({
left: rootLeft + 'px',
top: 0
});
curState = 'fixed';
}
}
function bottom() {
if (curState !== 'bottom') {
$root.removeClass('fixed').css({
left: 0,
top: rootBottomTop + 'px'
});
curState = 'bottom';
}
}
function setState() {
var scrollTop = $scrollTarget.scrollTop();
if (scrollTop >= rootTop && scrollTop <= scrollBottom) {
fixed();
} else if (scrollTop < rootTop) {
top();
} else {
bottom();
}
}
function init() {
if(!hasInit) {
var interval, timeout;
calc(true); setState();
// run calc every 100 millisecond
interval = setInterval(function() {
calc();
}, 100);
timeout = setTimeout(function() {
clearInterval(interval);
}, 45000);
window.pageLoad.then(function() {
setTimeout(function() {
clearInterval(interval);
clearTimeout(timeout);
}, 3000);
});
$scrollTarget.on('scroll', function() {
disabled || setState();
});
$window.on('resize', function() {
disabled || (calc(true), setState());
});
hasInit = true;
}
}
setOptions(options);
if (!disabled) {
init();
}
$window.on('resize', window.throttle(function() {
init();
}, 200));
return {
setOptions: setOptions,
refresh: function() {
calc(true, { animation: false }); setState();
}
};
}
$.fn.affix = affix;
});
})();
(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
function toc(options) {
var $root = this, $window = $(window), $scrollTarget, $scroller, $tocUl = $('<ul class="toc toc--ellipsis"></ul>'), $tocLi, $headings, $activeLast, $activeCur,
selectors = 'h1,h2,h3', container = 'body', scrollTarget = window, scroller = 'html, body', disabled = false,
headingsPos, scrolling = false, hasRendered = false, hasInit = false;
function setOptions(options) {
var _options = options || {};
_options.selectors && (selectors = _options.selectors);
_options.container && (container = _options.container);
_options.scrollTarget && (scrollTarget = _options.scrollTarget);
_options.scroller && (scroller = _options.scroller);
_options.disabled !== undefined && (disabled = _options.disabled);
$headings = $(container).find(selectors).filter('[id]');
$scrollTarget = $(scrollTarget);
$scroller = $(scroller);
}
function calc() {
headingsPos = [];
$headings.each(function() {
headingsPos.push(Math.floor($(this).position().top));
});
}
function setState(element, disabled) {
var scrollTop = $scrollTarget.scrollTop(), i;
if (disabled || !headingsPos || headingsPos.length < 1) { return; }
if (element) {
$activeCur = element;
} else {
for (i = 0; i < headingsPos.length; i++) {
if (scrollTop >= headingsPos[i]) {
$activeCur = $tocLi.eq(i);
} else {
$activeCur || ($activeCur = $tocLi.eq(i));
break;
}
}
}
$activeLast && $activeLast.removeClass('active');
($activeLast = $activeCur).addClass('active');
}
function render() {
if(!hasRendered) {
$root.append($tocUl);
$headings.each(function() {
var $this = $(this);
$tocUl.append($('<li></li>').addClass('toc-' + $this.prop('tagName').toLowerCase())
.append($('<a></a>').text($this.text()).attr('href', '#' + $this.prop('id'))));
});
$tocLi = $tocUl.children('li');
$tocUl.on('click', 'a', function(e) {
e.preventDefault();
var $this = $(this);
scrolling = true;
setState($this.parent());
$scroller.scrollToAnchor($this.attr('href'), 400, function() {
scrolling = false;
});
});
}
hasRendered = true;
}
function init() {
var interval, timeout;
if(!hasInit) {
render(); calc(); setState(null, scrolling);
// run calc every 100 millisecond
interval = setInterval(function() {
calc();
}, 100);
timeout = setTimeout(function() {
clearInterval(interval);
}, 45000);
window.pageLoad.then(function() {
setTimeout(function() {
clearInterval(interval);
clearTimeout(timeout);
}, 3000);
});
$scrollTarget.on('scroll', function() {
disabled || setState(null, scrolling);
});
$window.on('resize', window.throttle(function() {
if (!disabled) {
render(); calc(); setState(null, scrolling);
}
}, 100));
}
hasInit = true;
}
setOptions(options);
if (!disabled) {
init();
}
$window.on('resize', window.throttle(function() {
init();
}, 200));
return {
setOptions: setOptions
};
}
$.fn.toc = toc;
});
})();
/*(function () {
})();*/
</script><script>
/* toc must before affix, since affix need to konw toc' height. */(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
var TOC_SELECTOR = window.TEXT_VARIABLES.site.toc.selectors;
window.Lazyload.js(SOURCES.jquery, function() {
var $window = $(window);
var $articleContent = $('.js-article-content');
var $tocRoot = $('.js-toc-root'), $col2 = $('.js-col-aside');
var toc;
var tocDisabled = false;
var hasSidebar = $('.js-page-root').hasClass('layout--page--sidebar');
var hasToc = $articleContent.find(TOC_SELECTOR).length > 0;
function disabled() {
return $col2.css('display') === 'none' || !hasToc;
}
tocDisabled = disabled();
toc = $tocRoot.toc({
selectors: TOC_SELECTOR,
container: $articleContent,
scrollTarget: hasSidebar ? '.js-page-main' : null,
scroller: hasSidebar ? '.js-page-main' : null,
disabled: tocDisabled
});
$window.on('resize', window.throttle(function() {
tocDisabled = disabled();
toc && toc.setOptions({
disabled: tocDisabled
});
}, 100));
});
})();
(function() {
var SOURCES = window.TEXT_VARIABLES.sources;
window.Lazyload.js(SOURCES.jquery, function() {
var $window = $(window), $pageFooter = $('.js-page-footer');
var $pageAside = $('.js-page-aside');
var affix;
var tocDisabled = false;
var hasSidebar = $('.js-page-root').hasClass('layout--page--sidebar');
affix = $pageAside.affix({
offsetBottom: $pageFooter.outerHeight(),
scrollTarget: hasSidebar ? '.js-page-main' : null,
scroller: hasSidebar ? '.js-page-main' : null,
scroll: hasSidebar ? $('.js-page-main').children() : null,
disabled: tocDisabled
});
$window.on('resize', window.throttle(function() {
affix && affix.setOptions({
disabled: tocDisabled
});
}, 100));
window.pageAsideAffix = affix;
});
})();
</script><!---->
</div>
<script>(function () {
var $root = document.getElementsByClassName('root')[0];
if (window.hasEvent('touchstart')) {
$root.dataset.isTouch = true;
document.addEventListener('touchstart', function(){}, false);
}
})();
</script>
</body>
</html>