คำถามที่พบบ่อยเกี่ยวกับ SmooshGate

เกิดอะไรขึ้นกับการรวม

ข้อเสนอสำหรับฟีเจอร์ภาษา JavaScript ที่ชื่อ Array.prototype.flatten ใช้งานร่วมกับเว็บไม่ได้ การนำฟีเจอร์นี้มาใช้ใน Firefox Nightly ทำให้เว็บไซต์ยอดนิยมอย่างน้อย 1 เว็บไซต์ใช้งานไม่ได้ เนื่องจากโค้ดที่เป็นปัญหาเป็นส่วนหนึ่งของไลบรารี MooTools ที่แพร่หลาย จึงมีความเป็นไปได้ว่าเว็บไซต์จำนวนมากจะได้รับผลกระทบ (แม้ว่า MooTools จะไม่นิยมใช้กับเว็บไซต์ใหม่ในปี 2018 แต่เดิมแล้ว MooTools เป็นที่นิยมอย่างมากและยังคงมีอยู่ในเว็บไซต์เวอร์ชันที่ใช้งานจริงหลายแห่ง)

ผู้เขียนข้อเสนอแนะนำอย่างติดตลกให้เปลี่ยนชื่อ flatten เป็น smoosh เพื่อหลีกเลี่ยงปัญหาความเข้ากันได้ แต่หลายคนไม่เข้าใจว่าเป็นเรื่องตลก บางคนเริ่มเข้าใจผิดว่าชื่อใหม่ได้ถูกเลือกแล้ว และเรื่องก็เริ่มบานปลายอย่างรวดเร็ว

Array.prototype.flatten มีหน้าที่อะไร

Array.prototype.flat ซึ่งเดิมเรียกว่า Array.prototype.flatten จะยุบอาร์เรย์แบบซ้ำซ้อนจนถึง depth ที่ระบุ ซึ่งค่าเริ่มต้นคือ 1

// Flatten one level:
const array = [1, [2, [3]]];
array.flat();
// → [1, 2, [3]]

// Flatten recursively until the array contains no more nested arrays:
array.flat(Infinity);
// → [1, 2, 3]

ข้อเสนอเดียวกันนี้ยังมี Array.prototype.flatMap ซึ่งคล้ายกับ Array.prototype.map ยกเว้นว่าจะแปลงผลลัพธ์ให้เป็นอาร์เรย์ใหม่

[2, 3, 4].flatMap((x) => [x, x * 2]);
// → [2, 4, 3, 6, 4, 8]

MooTools กำลังทำอะไรที่ทำให้เกิดปัญหานี้

MooTools ระบุ Array.prototype.flatten เวอร์ชันที่ไม่ได้มาตรฐานของตัวเองดังนี้

Array.prototype.flatten = /* non-standard implementation */;

การใช้งาน flatten ของ MooTools แตกต่างจากมาตรฐานที่เสนอ แต่นี่ไม่ใช่ปัญหา เมื่อเบราว์เซอร์มีArray.prototype.flattenการใช้งานแบบดั้งเดิม MooTools จะลบล้างการใช้งานแบบดั้งเดิม วิธีนี้ช่วยให้มั่นใจได้ว่าโค้ดที่อาศัยลักษณะการทํางานของ MooTools จะทํางานตามที่ตั้งใจไว้ ไม่ว่า flatten เนทีฟจะพร้อมใช้งานหรือไม่ก็ตาม จนถึงตอนนี้ทุกอย่างเรียบร้อยดี

ขออภัย อาจมีบางอย่างเกิดขึ้น MooTools จะคัดลอกเมธอดอาร์เรย์ที่กำหนดเองทั้งหมดไปยัง Elements.prototype (โดยที่ Elements เป็น API สำหรับ MooTools โดยเฉพาะ)

for (var key in Array.prototype) {
  Elements.prototype[key] = Array.prototype[key];
}

for-in วนซ้ำพร็อพเพอร์ตี้ "ที่นับได้" ซึ่งไม่รวมเมธอดเดิม เช่น Array.prototype.sort แต่รวมพร็อพเพอร์ตี้ที่กำหนดเป็นประจำ เช่น Array.prototype.foo = whatever แต่และนี่ก็คือตัวดึงดูดความสนใจ หากคุณเขียนทับพร็อพเพอร์ตี้ที่แจกแจงไม่ได้ เช่น Array.prototype.sort = whatever ก็จะยังคงไม่สามารถแจกแจงได้

ปัจจุบัน Array.prototype.flatten = mooToolsFlattenImplementation จะสร้างพร็อพเพอร์ตี้ flatten ที่นับได้ ดังนั้นระบบจึงคัดลอกไปยัง Elements ในภายหลัง แต่หากเบราว์เซอร์จัดส่ง flatten เวอร์ชันเนทีฟ flatten จะไม่สามารถแจกแจงได้ และจะไม่คัดลอกไปยัง Elements โค้ดใดก็ตามที่อาศัย Elements.prototype.flatten ของ MooTools จะไม่ทำงาน

แม้ว่าการเปลี่ยน Array.prototype.flatten เดิมให้เป็นแบบที่เรียกข้อมูลได้จะช่วยแก้ปัญหาได้ แต่ก็มีแนวโน้มที่จะทำให้เกิดปัญหาความเข้ากันได้มากขึ้น เว็บไซต์ทุกแห่งที่ใช้ for-in เพื่อวนซ้ำผ่านอาร์เรย์ (ซึ่งเป็นแนวทางปฏิบัติที่ไม่ถูกต้อง แต่เกิดขึ้นได้) จะได้รับการวนซ้ำของลูปเพิ่มเติมสำหรับพร็อพเพอร์ตี้ flatten อย่างกะทันหัน

ปัญหาที่ใหญ่กว่าคือ การแก้ไขออบเจ็กต์ในตัว ปัจจุบันการขยายโปรโตไทป์แบบเนทีฟถือเป็นแนวทางปฏิบัติที่ไม่ถูกต้อง เนื่องจากใช้งานร่วมกับไลบรารีอื่นๆ และโค้ดของบุคคลที่สามได้ไม่ดี อย่าแก้ไขวัตถุที่คุณไม่ได้เป็นเจ้าของ

ทำไมเราไม่เก็บชื่อเดิมไว้และทำลายเว็บไปเลย

ในปี 1996 ก่อนที่ CSS จะแพร่หลาย และก่อนที่ "HTML5" จะกลายเป็นที่นิยม เว็บไซต์ Space Jam ได้เปิดตัว ปัจจุบันเว็บไซต์ดังกล่าวยังคงทำงานเหมือนเดิมเหมือนเมื่อ 22 ปีก่อน

เกิดขึ้นได้อย่างไร มีเจ้าหน้าที่ดูแลเว็บไซต์ดังกล่าวตลอดหลายปีที่ผ่านมาไหม โดยอัปเดตทุกครั้งที่ผู้ให้บริการเบราว์เซอร์เปิดตัวฟีเจอร์ใหม่

ผลที่ได้คือ "อย่าแตกเว็บ" เป็นหลักการออกแบบอันดับ 1 สำหรับ HTML, CSS, JavaScript และมาตรฐานอื่นๆ ที่ใช้กันอย่างแพร่หลายบนเว็บ หากการเปิดตัวฟีเจอร์ใหม่ในเบราว์เซอร์ทําให้เว็บไซต์ที่มีอยู่หยุดทํางาน สิ่งต่างๆ จะส่งผลเสียต่อทุกคนดังนี้

  • ผู้เข้าชมเว็บไซต์ที่ได้รับผลกระทบได้รับประสบการณ์การใช้งานที่ใช้งานไม่ได้อย่างกะทันหัน
  • เจ้าของเว็บไซต์มีเว็บไซต์ที่ทํางานได้อย่างสมบูรณ์แบบแต่กลับใช้งานไม่ได้โดยที่เจ้าของไม่ได้เปลี่ยนแปลงอะไรเลย
  • ผู้ให้บริการเบราว์เซอร์ที่เปิดตัวฟีเจอร์ใหม่จะสูญเสียส่วนแบ่งการตลาดเนื่องจากผู้ใช้เปลี่ยนไปใช้เบราว์เซอร์อื่นหลังจากสังเกตเห็นว่า "ฟีเจอร์นี้ใช้งานได้ในเบราว์เซอร์ X"
  • เมื่อทราบปัญหาด้านความเข้ากันได้แล้ว ผู้ให้บริการเบราว์เซอร์รายอื่นๆ ปฏิเสธที่จะจัดส่ง ข้อกำหนดของฟีเจอร์ไม่ตรงกับความเป็นจริง ("เป็นเพียงเรื่องแต่ง") ซึ่งส่งผลเสียต่อกระบวนการมาตรฐาน

แน่นอนว่าเมื่อมองย้อนกลับไป MooTools ได้ทำสิ่งที่ไม่ถูกต้อง แต่การทำให้เว็บใช้งานไม่ได้ไม่ได้เป็นการลงโทษ MooTools แต่เป็นการลงโทษผู้ใช้ ผู้ใช้เหล่านี้ไม่ทราบว่า Moot Tool คืออะไร หรือเราจะหาวิธีแก้ปัญหาอื่นก็ได้ และผู้ใช้ก็ใช้เว็บต่อไปได้ การเลือกนั้นง่ายมาก

Does that mean bad APIs can never be removed from the Web Platform?

แล้วแต่กรณี ในบางกรณีที่เกิดขึ้นไม่บ่อยนัก ระบบอาจนำฟีเจอร์ที่เป็นอันตรายออกจากเว็บ แม้แต่การระบุว่าเป็นไปได้หรือไม่ที่จะนำฟีเจอร์ออกก็เป็นเรื่องที่ซับซ้อนมาก และต้องอาศัยการวัดผลที่ครอบคลุมเพื่อประเมินจำนวนหน้าเว็บที่จะมีการเปลี่ยนแปลงลักษณะการทำงาน แต่เมื่อฟีเจอร์นี้ไม่ปลอดภัยพอ เป็นอันตรายต่อผู้ใช้ หรือมีการใช้งานน้อยมาก ก็อาจเป็นไปได้

<applet>, <keygen> และ showModalDialog() ล้วนเป็นตัวอย่างของ API ที่ไม่ถูกต้องซึ่งนําออกจากแพลตฟอร์มเว็บได้สําเร็จ

ทำไมไม่แก้ไข MooTools เลย

การแพตช์ MooTools เพื่อไม่ให้ขยายออบเจ็กต์ในตัวอีกต่อไปเป็นความคิดที่ดี แต่ก็ไม่สามารถแก้ไขปัญหาเฉพาะหน้าได้ แม้ว่า MooTools จะเผยแพร่เวอร์ชันแพตช์ แต่เว็บไซต์ที่มีอยู่ทั้งหมดซึ่งใช้เวอร์ชันนี้จะต้องอัปเดตเพื่อให้ปัญหาความเข้ากันได้หายไป

ผู้ใช้อัปเดต MooTools ที่ใช้อยู่ไม่ได้หรือ

หากทุกอย่างเป็นไปตามที่ควรจะเป็น MooTools จะปล่อยแพตช์ และเว็บไซต์ทุกแห่งที่ใช้ MooTools จะอัปเดตโดยอัตโนมัติในวันถัดไป ปัญหาได้รับการแก้ไขแล้ว ถูกต้องไหม

ขออภัย การดำเนินการนี้ไม่สามารถทำได้ แม้ว่าจะมีผู้ระบุเว็บไซต์ทั้งหมดที่ได้รับผลกระทบ ค้นหาข้อมูลติดต่อของเว็บไซต์แต่ละแห่ง ติดต่อเจ้าของเว็บไซต์ทั้งหมดได้สำเร็จ และโน้มน้าวให้เจ้าของเว็บไซต์ทั้งหมดทำการอัปเดต (ซึ่งอาจหมายถึงการปรับโครงสร้างฐานโค้ดทั้งหมด) แต่กระบวนการทั้งหมดนี้อาจใช้เวลาหลายปีเป็นอย่างดีที่สุด

โปรดทราบว่าเว็บไซต์เหล่านี้จำนวนมากเป็นเว็บไซต์เก่าและน่าจะไม่มีการบำรุงรักษา แม้ว่าผู้ดูแลยังอยู่กับคุณ แต่ก็อาจเป็นไปได้ว่าไม่ใช่นักพัฒนาเว็บที่มีทักษะสูงเหมือนคุณ เราไม่อาจคาดหวังได้ว่าทุกคนจะเปลี่ยนเว็บไซต์อายุ 8 ขวบของตน เนื่องจากปัญหาเรื่องความเข้ากันได้ของเว็บ

กระบวนการของ TC39 ทำงานอย่างไร

TC39 เป็นคณะกรรมการที่รับผิดชอบการพัฒนาภาษา JavaScript ผ่านมาตรฐาน ECMAScript

#SmooshGate ทำให้บางคนเชื่อว่า "TC39 ต้องการเปลี่ยนชื่อ flatten เป็น smoosh" แต่จริงๆ แล้วเป็นมุกตลกภายในที่สื่อสารกับภายนอกได้ไม่ชัดเจน การตัดสินใจที่สำคัญ เช่น การเปลี่ยนชื่อโปรเจ็กต์ ไม่ใช่การตัดสินใจที่ดำเนินการโดยพลการ ไม่ใช่การตัดสินใจที่ดำเนินการโดยบุคคลเพียงคนเดียว และไม่ใช่การตัดสินใจที่ดำเนินการในชั่วข้ามคืนโดยอิงตามความคิดเห็นเดียวใน GitHub

TC39 ดำเนินการตามกระบวนการเตรียมความพร้อมที่ชัดเจนสำหรับข้อเสนอฟีเจอร์ ข้อเสนอ ECMAScript และการเปลี่ยนแปลงที่สำคัญทั้งหมด (รวมถึงการเปลี่ยนชื่อเมธอด) จะได้รับการพูดคุยกันในการประชุม TC39 และต้องได้รับอนุมัติจากคณะกรรมการทั้งหมดก่อนที่จะมีผลอย่างเป็นทางการ ในกรณีของ Array.prototype.flatten ข้อเสนอผ่านขั้นตอนต่างๆ ของข้อตกลงแล้วจนถึงระยะที่ 3 ซึ่งบ่งบอกว่าฟีเจอร์พร้อมใช้งานในเว็บเบราว์เซอร์ ปัญหาเกี่ยวกับข้อกําหนดเพิ่มเติมเกิดขึ้นได้ในระหว่างการติดตั้งใช้งาน ในกรณีนี้ ความคิดเห็นที่สําคัญที่สุดคือความคิดเห็นที่ได้รับหลังจากพยายามเปิดตัวฟีเจอร์ดังกล่าว เนื่องจากฟีเจอร์ดังกล่าวทำให้เว็บใช้งานไม่ได้ในสถานะปัจจุบัน ปัญหาที่คาดการณ์ได้ยากเช่นนี้เป็นหนึ่งในเหตุผลที่กระบวนการ TC39 ไม่ได้สิ้นสุดลงเมื่อเบราว์เซอร์เปิดตัวฟีเจอร์

TC39 ทำงานแบบฉันทามติ ซึ่งหมายความว่าคณะกรรมการต้องเห็นด้วยกับการเปลี่ยนแปลงใหม่ แม้ว่า smoosh จะเป็นคำแนะนำที่จริงจัง แต่ดูเหมือนว่าสมาชิกคณะกรรมการจะคัดค้านชื่อนี้และต้องการชื่อที่พบได้ทั่วไปมากกว่า เช่น compact หรือ chain

การเปลี่ยนชื่อจาก flatten เป็น smoosh (แม้ว่าจะไม่ใช่การพูดเล่น) ไม่เคยมีการพูดคุยกันในการประชุม TC39 ดังนั้น ขณะนี้เรายังไม่ทราบจุดยืนอย่างเป็นทางการของ TC39 เกี่ยวกับหัวข้อนี้ ไม่มีบุคคลใดสามารถพูดในนามของ TC39 ทุกคนจนกว่าจะมีความเห็นพ้องกันในการประชุมครั้งถัดไป

โดยทั่วไปแล้ว การประชุม TC39 จะมีผู้เข้าร่วมที่มาจากหลากหลายภูมิหลัง เช่น บางคนมีประสบการณ์การออกแบบภาษาโปรแกรมมาหลายปี บางคนทำงานเกี่ยวกับเบราว์เซอร์หรือเครื่องมือ JavaScript และมีผู้เข้าร่วมจำนวนมากขึ้นที่เข้าร่วมเพื่อแสดงถึงชุมชนนักพัฒนาซอฟต์แวร์ JavaScript

SmooshGate ได้รับการแก้ไขแล้วหรือยัง

ในช่วงการประชุม TC39 ของเดือนพฤษภาคม 2018 ปัญหา #SmooshGate ได้ถูกแก้ไขอย่างเป็นทางการด้วยการเปลี่ยนชื่อ flatten เป็น flat

Array.prototype.flat และ Array.prototype.flatMap พร้อมใช้งานใน V8 v6.9 และ Chrome 69