I have adopted many ECMA innovations quickly and happily. One that I have not is the new class sugar. To me, it violates JS clarity.I have never been a fan of "new thing()" because it creates a brand new "this". In my view, that is a weird global variable that is ripe for abuse and bugs, especially so since all of its properties are, by definition, mutable. Worse, for me, properties of this are exposed to the outside world. Preventing that requires additional code that does not contribute to the actual functionality.
When arrow functions showed up and gave us control over our 'this' and closure context, I devised the pattern of using a function initializer. It replaces properties of 'this' with closure variables that never can leak to the outside world by accident. Internal values can be mutable or not using normal 'let' and 'const'.
It is true that I absolutely never use inheritance in other languages that I use (PHP and C#). I have drunk "composition over inheritance" koolaid to the point of addiction, or devotion, or certainty.
So I ask, what am I missing? Should I be using classes? It seems a lot of people prefer it but I don't understand why.
Your thoughts, observations, comments?
In the the repository directory...
1) Do git log to show, then copy commits done since that last push.
2) Delete (only) the .git folder (rm -rf).
Then...
1) Clone the repo from GitHub in a temporary directory.
2) Move the newly cloned .git folder from the temporary directory into the repository directory.
3) Do a commit. (I included the commit messages from the log copy made above, plus an explanation of the commit message discontinuity.)
4) Push.
QED
---------------------------------------------
Jina, Fix this up.
Firstly, within the repository directory:
1) Execute `git log` to display and then copy the commits done since the last push.
2) Remove only the .git folder using `rm -rf .git`.
Subsequently:
1) In a temporary directory, clone the repository from GitHub.
2) Transfer the newly cloned .git folder from the temporary directory to the repository directory.
3) Perform a commit. This should include the commit messages copied earlier, along with an explanation for the discontinuity in the commit messages.
4) Push your changes.
And that's it!
]]>I found way too many explanations for how to configure Sonicwall to forward a an external port to an internal destination. Everyone seems intent on trying to explain too much until I can to this old (2017) one by Sonicwall's M. G. Sriram Iyer. It's like a dream come true, just the specifics of creating the capability. I took notes.
My specific need was to forward traffic on port 9000 to port 9000 on a specific internal system.
Create address object for internal and external addresses
[on my system, this already exists as Default WAN]
[eg, New Internal Address Object points to the internal IP address]
create service definition assigning TCP (or whatever) to the internal port number
[eg, New Internal Port Assignment Service points to port 9000]
create firewall->access to allow traffic, WAN/LAN
service: http
source: any
dest address: address object (eg, Default WAN)
dest service: service object (New Internal Port Assignment Service)
_must be above (priority) any deny rule_
create NAT Policy to translate and forward traffic
orig source: any
orig dest: an address object (Default WAN)
orig service: http
trans source: origin
trans dest: an address object (New Internal Address Object)
trans service: service object (New Internal Port Assignment Service)
inbound interface: X1
So, when I found that, as I implement a new app, that OpenAI has made it's new, V4, npm module entirely incompatible with my previous CommonJS library, I was annoyed that I could not just pick up previous code. When I found that it is only published as ESM and that you can't just put "import openAi from openai" inline with require(), I was furious.
Trying to figure out how to use ESM in a CJS world, I have read so many idiots saying, "Don't. ESM is great.
Rewrite your code base to be cool and modern." Or, convoluted crap that
tells you how to do some complicated thing related to this topic. I have
to tell you that my last couple of hours have been an epic fail in
Google, StackOverflow and everyone else.
It made me so mad that I was
forced to the final resort: RTFM. In this case, the NodeJS module documentation.
I'm sure you have as little interest in reading about ESM as I do, so ere is the answer:
I wrote an interface module. All it does is use the NodeJS import() function with its promise. (You can obviously use this without the structure but I personally prefer callbacks to async/await and think this is easier to interpret.)
This module only import()'s and is called mport-openai.js:
'use strict';
module.exports = async function(callback) {
const openAi = await import('openai');
callback('', openAi);
};
This is in my main program:
require('./import-openai')((err, openAi)=>{
//do cool stuff here
});
I make no claim that this is brilliant insight. In fact, I know it is totally obvious... Once you know about it,which you won't until you do and Google was no help until you got here.
You're welcome.
ps, You might have thought, "Why'd he have to tell me this whole long story about legacy CommonJS, ESM, node modules, looking for import(), etc?" The answer is Google Bait, not that I think you actually care about my journey. It needs the explanation for the index.
I go for long periods of time where I know the answer to this but then, brain fart, I get confused. I have been amazed at how hard it is to find a clear statement about this. Now there is one. You're welcome.
If I want to merge source files into a target directory, I should not use a slash on the target directory path..
If I want to put a directory inside of a target directory, I should use a final slash on the target directory path.
Repeat after me: Final slash means the new directory. No slash means merge.
sourceDir
fileOne
fileTwo
targetDir
someFileOne
someFileTwo
I get a merge If I use:
cp -R sourceDir targetDir # no final slash
My targetDir ends up looking like:
targetDir
someFileOne
someFileTwo
fileOne
fileTwo
I get a new subdirectory if I add a final slash, eg:
cp -R sourceDir targetDir/ # yes final slash
The result is:
targetDir
someFileOne
someFileTwo
sourceDir
(And targetDir/sourceDir contains fileOne, fileTwo.)
Repeat after me: No slash to merge. Yes slash to create a new subdirectory.
The javascript phrase
test`That ${person} is a ${age}.`
Has never been useful to me and so I have never really understood it. Today I found something that relies on it. As a consequence, I had to figure out how it works. The usually awesome MDN gave working examples but did a terrible job of explaining what was going on. I hereby leap into the breach.
The first thing is that the syntax is a sort of function call with no parenthesis. It is a function name followed immediately by a literal. Weird to look at but there you have it.
The next thing is that the function execution has a big, complicated process behind it. When it is called, the JS engine does these things...
Eg,
The effective function call ,
test(stringsArray, person, age));
This allows you to process the components of the literal in any way you want. Assemble the strings backwards? Sure. Put the values in the wrong place? Can do.
Below I constructed an example that acts as if there was no function. I felt like it helped me understand.
I suffer a great deal of hassle trying to deal with the insane amount of details that come along with being a programmer. Partly because I am really old, I cannot remember a single thing but really, I have not been able to ever remember details. I have long developed practices to accommodate this weakness.
I started by using BASH aliases to remember various commands. I still have a couple of aliases from fifteen years ago for lsof, a command I use rarely and can never remember how to work. Later, I got the idea of having an alias (initProjA) always present in my shell for each project that executed a script, initDevTerminal, that was in a management directory I kept in every project. This script created aliases and also showed notes for things I might forget. EG, I keep a project for my computer that just tells me where the NGINX config files are located since I can never remember.
For real projects, the initDevTerminal script generates aliases that initialize or execute tests, copy code to production servers, execute the toolchain, start and stop things, whatever is needed. Some projects have a dozen entries. Some fewer. One longstanding and stupidly complicated one actually has pages for the various subsections. The script also initializes environment variables if needed and, in some cases, swaps out launchctl or systemctl jobs. The important thing is that most of the complicated commands are put into aliases or listed as notes when 'initProjA' is executed on the command line for easy reference.
I also manage applications on various client servers. For a long time, I would put such a helper script in my BASH environment on the client systems. Over the years, that became complicated to manage so I changed the structure. Now I put the scrips in the system specific config directory, the one where my programs look to find out, eg, the actual path or API key or something.
In the directory for each computer, my development one include, there is a directory called terminalAndOperation. Each has a well-known file named 'initTerminal'. Depending on when I last worked on the project, that file might be an old, simple one or a new, cool one that includes, for example, boilerplate reference to a common file at the root of the configs directory so I do not have to repeat things that are, well, common to all the environments.
The main things I have figured out are, 1) use a script to contain all the stuff I know I will not remember, 2) structure projects so that there is a script for each environment I have to work on, 3) be rigorous in making all that structure the same because I know I will not remember where the script is if it's different, 4) use aliases (also, btw, listed in my .bash_profile) to execute them so I am not confused and 5) make sure to write all the fussy details in those scripts and keep them up to date and refactored frequently.
This has allowed me to keep being a productive programmer long into senility. My wife has to remind me about everything but, when my colleagues need to know fussy details, they ask me and I can easily find the answer.
(Yes, this looks familiar. It's the same debugging session as 'runs when started'. Be patient for heaven's sake.)
I grabbed some boilerplate off the web to make my systemd timer/service pair work. Worked fine.
ie,
systemctl enable myThing.timer
systemctl enable myThing.service
systemctl start myThing.timer
#do not start myThing. service or it will run immediately, that's ok at some other time for a special non-timer purpose.
I started my debugging with a quick script that just logged hello and it all worked quite nicely.
What I didn't notice initially is that it was running the target service when I started the timer.
Later, I changed it to refer to my real, long-running (hours) process and, as you might imagine, that made it very clear that it was being run an extra time.
The problem is that my boilerplate had a 'Requires' statement in the Unit stanza. When this is around, systemd does a 'start' on that Require'd process when the timer is started.
The Require statement I copied said...
Requires=myThing.service (just like it had Unit=myThing.service in the Timer stanza)
Turns out that systemd starts the Require'd service if it is not already running in case, I suppose, you forgot to.
The only reason I can imagine having the target process in the Require statement is so that it runs once at startup. I don't know but, there you have it. If you are getting extra invocations of your service, get it out of the Require statement.
I grabbed some boilerplate off the web to make my systemd timer/service pair work. Worked fine.
ie,
systemctl enable myThing.timer
systemctl enable myThing.service
systemctl start myThing.timer
#do not start myThing. service or it will run immediately, that's ok at some other time for a special non-timer purpose.
Showed a nice schedule with 'status' and it ran correctly at the right time.I did some debugging and experiments to tune the system and, after seeming to work ok, it started telling me that the timer was dead. I could find no reason for a long, harrowing debug session.
Here is the answer.
I had started the process by implementing the timer/service pair pointing at a trivial test script. It ran and quit instantly. I did not notice that it was running and quitting on timer start. I also think that I (incorrectly) did "systemctl start" on the quick test service which then exited normally.
Later, I wired it up to my real process, one that takes more than an hour to execute. Of course, I don't want to wait that long so I did "systemctl stop".
After that, the timer would not work. It said it was dead. I tried everything.
The problem is that the boilerplate I grabbed included a "Requires" statement in the Unit stanza. It was set to the same name (myThing.service) as the target service so I did it too. That was an error. I do not know why the boilerplate author included it but requiring the target.service as a dependency makes no sense even though it basically worked.
When I did systemctl stop the long-running target service, that put it into a state where the dependency was not able to be met. Not entirely sure what the difference between never run and stop is but, I have proven this.
When I removed the extraneous 'Require' statement, that dependency went away, the timer started correctly and all was happiness and unicorns.
What I still don't know (Hey, Smartypants, that's what comments are for) is why stopping the target service made it violate the Require constraint.
We are using a locally hosted instance of gitlab. It's a nice program and are enjoying many things about it.
Most of the people working on it live a fairly simple life. I am not one of them. I work a zillion servers and SSH all over the place. I have many RSA keys for various purposes. To keep track of them, I used the comment section of the public key to let me know which one is which.
EG, a public key might end
...SSCtQUvSJ5jdfW9YB3w== mySpecialKeyName
Time came to move a repo from github to gitlab. I grabbed the public key I was using for github and pasted it into the settings of gitlab. It accepted it happily.
Also, it did not work. When I tried to push my newly re-origined ("git remote add origin git@...") repo to git lab, it asked me for a password. There is, of course, no password nor should it ever ask for one.
After trying EVERYTHING, I noticed instructions in the gitlab page that reminded me to copy my public key completely. It said from ssh-rsa to the "email address".
I changed it to
...SCtQUvSJ5jdfW9YB3w== tq@myGitlabEmailAddress.com
And it worked.
You're welcome.
PS, Gitlab, if you ever hear about this: This is bullshit. There is not a single annotation anywhere. Even if I was in the habit of using the RSA key comment section for an email address, there's no reason to imagine it would be the right email address. If you're not going to make *me* responsible for establishing the connection between the key and my account, at least do me the solid of giving me an error message when it doesn't have the right info.
Obviously, nobody wants to use Apple's Terminal in a world where there is iTerm2.
It's equally obvious that Barebones bbedit is awesome.
If you want the context menu selection, Go Here in Terminal to open in iTerm2, paste this into the command line:
defaults write com.barebones.bbedit TerminalBundleID "com.googlecode.iterm2"
(Make sure the quotes weren't made curly by your web browser. You can use bbedit->Text->Straighten Quotes to make sure.)
I've been going crazy. I need to change my company email signature. I do so, it works fine but next time I quit Mail.app it goes away. I messed around with plists and permissions. Nothing helped.
Then I found an article HERE. It had the answer. I repeat it here for posterity.
The problem arises from some bug between Mail.app and iCloud Drive. The problem has been around at least since 2017. There does not appear to be a fix that will make Mail.app save signatures as long as it is involved with iCloud Drive. Fortunately, you can turn that on and off with no apparent harm.
Go to System Preferences->iCloud Drive, then click on iCloud Drives Options... button. In there, you will see a list that includes Mail. Uncheck Mail.
(Note: Mail appears in the list initially shown when you open the iCloud preference panel. This is not the correct one to uncheck. You need to click into iCloud Drive Options and uncheck Mail there.)
Back in Mail.app, you will find that signatures save correctly. Do your work. Quit Mail and reopen and you will find you new and changed signatures in place. Win!!
Once done, you can return to the iCloud Preference Panel->iCloud Drive->options and recheck Mail. Signatures will remain. Life will be good.
The context is that I had a request to add a feature to a Nodejs app that I wrote a couple of years ago. The purpose of the app is to grab a bunch of files off of an SFTP server and make files out of them. The purpose of this post is to provide a google friendly summary so nobody has to spend as much time on this as I did.
So, I fired up the app on my development machine. It looked ok at first but then it just stopped processing. Further investigation told me that the downloaded files got to disk and that the problem was that the ‘close’ event was never firing.
Inspiration strikes and I realize that the app was written for an older version of Nodejs. More experimentation revealed that the app runs correctly on versions up to 9.11.2 but at 10.0.0, it fails. For google, I restate: upgrade nodejs to version ten causes readable stream to never fire ‘close’ (or ‘end’ or any other) event.
After a full day and a half of research and experimentation, I learned that, starting in nodejs 10, readable streams default to ‘pause’ mode. Previously, they were in ‘flowing’ mode. In ‘pause’, the stream waits for the consumer to take data before it does anything else.
Well, sort of. It does provide the data, somehow. After all, my zip files were created and did work correctly.
In any case, I found a clue someplace that there had been a change to the default for streams. I found another clue that talked about the ‘paused’ behavior. Eventually I found a comment on stackoverflow (here) that mentioned read();
So, here’s the code:
sftpControls.sftpConnection
.get(remoteDirectoryPath + name, false, null)
.then(zipSourceStream => {
const filePath = `${zipFileDirPath}/${name}`;
zipSourceStream
.pipe(qtools.fs.createWriteStream(filePath, { encoding: null }))
.on('close', () => {
zipSourceStream.destroy();
next('', name);
});
zipSourceStream.read(); //tell the stream to be ‘flowing’
})
That is the entire change. Add one read() and the stream becomes ‘flowing’ and emits a close event when it’s done.
A small warning. I tried to chain the .read() along with the pipe() and on() methods. That failed. The read() method does not return the source object. You will get an error, “no read().pipe()”. Reverse it and get “no pipe().read()”. You have to run it like it’s shown, directly off of the original stream object.
If anyone reading this actually understands why this is happening or how it works, I am sure future programmers will be grateful to read the comments. I know I will.
tl;dr: You can't. At least not in Javascript. (I beg you. Prove me wrong in the comments!)
Here's the problem I face.
I have a template system I use. It relies on a .ini file that has all the parts of (in this case) an email message. Subject, text body, html body, etc. It replaces tags in the elements with properties of a supplied object of the same name. The template also includes a property called 'transformations'.
Transformations contains a set of properties whose values are functions. When processing the replacement, my system runs the transformations against the other replacement properties and adds the result for replacement in the template. It's useful for making sure that names are capitalized or adding dates or changes to html styling based on the data.
The new challenge is that I need to have a transformation that gets a result from the database. The problem is that the database access is asynchronous. The system is set up to run simple string processing synchronously. There is no callback option.
I spent an entire day screwing around with generators/yield, async/await and reading what felt like the entire internet looking for an idea of how to deal with this. I consider putting the database function into code, per se, to be a crime. (I do not mind calling a specific system function but it has to be called from the primary locus of control, ie, the template. I don't want someone to look up later and find that the result is just not there anymore with no clue where it should have come from.)
I ended up adding a property to the template called 'asyncronousTransformations' and having the program that uses the template process them. This required back filling asynchrony several levels up the code. As you might imagine, sending email is asynchronous but the (hitherto) simple string processing was not. Now it is.
My iPhone has been a huge pain in the butt since iPhone X made it so that it thought my nipple under my shirt pocket is a finger that required it to do whatever my nipple wanted to do.
No more!! This article explains, settings->general->accessibility->tap to wake!! Disable it and my nipple is no longer empowered to hang up my phone calls.
And, it's harmless because settings->display & brightness->raise to wake makes it so that it almost always automatically wakes up when I want it to.
Salvation.
read more: How to turn off the Tap to Wake feature on your iPhone X
This is almost trivial unless you forget about it and try to research the solution. It's stupidly obscure, probably because it's so trivial.
I am using request to post into some application. I didn't have a convenient boilerplate to reference and the damn internet did not provide a decent example. Here is a working and completely adequate post block:
request.post(
{
url:url,
headers: {
authorization: `${userId} ${authToken}`,
'Content-Type': 'application/json'
},
body: JSON.stringify({ propertyName:'propertyValue' })
},
(err, response, body) => {
console.dir({ 'err': err });
console.dir({ 'response': response });
console.dir({ 'body': body });
//callback(err, body);
}
);
This works but it didn't at first. On the receiving end (also nodejs, using express), the call was just fine, came back 200, showed its presence on the far end but, absolutely no post body content. The problem? I had forgotten to include the Content-Type header.
macOS sftp "no matching cipher found"
Add this to ~/.ssh/config
Host * SendEnv LANG LC_* Ciphers +aes256-cbc
Works like a charm.
Thanks to Jason.
read more: SSH issues with Mac OS X High Sierra
My interest in public key encryption continues. I wanted to be able to actually use a tool to play so I wrote one.
It does these things:
1) Generate a key pair.
2) Extract a public key from a private key.
3) Manually enter public or private key.
4) Create a crypto text string from plain text input.
5) Extract plain text from a crypto string.
You can play with this at: http://genericwhite.com/rsaEncryptionDemo/
Here's the problem. A co-worker got a bizarre alert() dialog in a web app. It popped up and just said "1". It wasn't her code and she was completely stumped. That's how I got involved.
I looked around the code. Searched for errant /alert\(/, etc. Nothing worked.
Then I hit the developer console. I typed the best single thing I ever typed:
window.alert= msg=>{ console.log(msg); console.trace(); }
Yeh, I did that. You may enjoy my awesomeness.
Console.trace() gives a stack dump at the location that the alert dialog was called. That was very helpful.
You are welcome.
Access for Learning (a4l.org) is happy to announce a major new addition to the School Interoperability Framework (SIF) data model to support students with special needs. Comprising two major components, xIndividualizedEducationPlan and xIepTransfer, this release is the result of a two year effort led by TQ White II of the Central Minnesota Educational Research and Development Corporation (cmerdc.org) with the help of national experts in special education and data modeling. The effort is motivated by the recognized need to make a student’s individualized education plan (IEP) content available when a student transfers into a new school.
The new data models are intended to support three main use cases. 1) Immediate support for an administrator the very first time a student shows up in a new school. 2) Information to support the special education team as they adapt plans already in place to the resources and strategies of the receiving school. 3) Sufficient information for schools and districts to support reporting and resource management needs. The goal is to ensure that a school has the information needed to provide students having special needs with critical, ongoing services.
This new model is based on a thorough survey of the standard form sets published by nearly every state, as well as the federal government. They were categorized into representative groups for an exhaustive inventory of data and evaluation of documentation strategies. With the input of a workgroup averaging about ten people a structured hierarchy of elements was developed and refined. Once done, the work was passed to Jill Parkes, education data analyst, at CEDS (Common Educational Data Standards) a federal organization that develops a dictionary of education related data definitions.
The CEDS process did two things. First, they evaluated each element in the new, tentative IEP data model and, where appropriate, attached a formal definition to it, either new or in reference to an existing definition. Then it was put into a formal CEDS community review. CEDS stakeholders, especially those with an interest in special education, reviewed the new definitions and approved them. This discussion improved confidence in the data design and made it more complete.
After this information was added to the XML – and with substantially more confidence, the data model was formally moved into the Access for Learning community review process. Though some people looked at the XML and offered comments, the main process involved TQ making presentations to various groups explaining the process and product in detail. Many valuable comments were made that resulted in changes but two were especially valuable.
First was Megan Gangl, a co-worker of TQ’s at cmERDC. Megan has spent her entire career as special education teacher, case manager and leader of case managers. Her decades of experience brought many new details to the model, suggested reorganization of some parts and validated others. She identified missing details, helped to rename elements and refine both their data definitions and the explanations of their meanings. After the initial presentation, she spent several days collaborating on the model in detail. Once done, confidence in the usefulness, correctness and completeness of the model was again tremendously improved.
The day before community review started in October, a new person, Danielle Norton, joined the North American Technical Board. Danielle’s team contributed to the community review with sessions including the detailed overview presentation and discussions with various subgroups of her team. A particularly important contribution was made by Rick Shafer, a long experienced data architect, who noted some problems with normalization in the data model.
The initial motivation for the IEP effort was to support the transfer of students between schools or districts. Throughout the process, the foremost intention was to provide complete information for the receiving educational agency. As a consequence, the data model included data elements that were duplicates of things that were defined elsewhere in SIF. That is, it was badly de-normalized. It made it so that the element would provide a complete picture for a receiving district but was ill-suited for use as a local SIF entity object.
To solve this problem, the data model was split into two elements, xIndividualizedEducationPlan and xTransferIep. The former is completely normalized to serve as a formal entity. No data is represented that is defined elsewhere in SIF but is, instead, referenced with a refId. If a receiving program needs to know those details, it is expected to query the appropriate system for details.
The latter is conceived as a reporting object, i.e., it is intended to wrap information that is defined elsewhere for convenient reference. The xTransferIep includes structures that allow it to contain data referenced in the IEP that would otherwise require a query to a system to which the receiving organization may not have access. The xTransferIep is a complete representation of an IEP containing all details.
In this process, a new concept was added to SIF, the typed refId. Troubled by the fact that refIds inside the IEP provided no information about where the target information referenced by the refId could be found, TQ added several new data types to the data model. Each is a UUID (as is the generic refId) but each also included documentation elements that explain what the UUID refers to and where the data can be found. For example, one of the new types, iepCommonStudentContactRefIdPointerType, explains that it references a contact inside a student object, distinct from iepCommonContactRefIdPointerType, which points to an independent xContactType, e.g., service provider or doctor, somewhere else.
The last thing is that, with the help of Access for Learning’s John Lovell, the new data models were refined to fit the new xPress object strategy. It does not use XML attributes and refIds are only present for elements that need it. This allows easier use of the model in non-Java/.NET systems. xPress is a more recent addition to SIF v3 and has proven to be easier to work with and, consequently, more popular. It is expected that xPress will be the foundation of new infrastructure work to formally bring JSON into the data model.
As with any first effort, it is fully understood by TQ and the entire community that as this data model comes into actual use, shortcomings will be noted and ideas will be conceived. It is intended that the SpecEd/IEP workgroup will reconvene in the future to evaluate the results of implementation. That is to say that, as with the rest of SIF, the new IEP data models being released with SIF v3.5 are not the end of the effort to better support students with special needs. This release is the beginning of an ongoing effort to insure that SIF is able to help schools, districts and teachers have the information needed to support optimal educational outcomes and to allow students with special needs to have the brightest possible future.
For even more information, a video recording of the IEP Data Model Overview is available HERE. To contact TQ White II, leave word in the comments.
]]>I don't reset mysql's root password often enough to remember how to do it. So, I google and go through endless hassle because all the examples are old or incomplete. It's maddening.
The main problem is that nobody includes the stuff below referring to /var/run/mysqld. I don't know why. Perhaps it was not needed in the past. However, it sure is now. You can tell if you do by seeing this:
mysqld_safe Directory '/var/run/mysqld' for UNIX socket file don't exists (2)
when you try 'mysql -uroot mysql' without it.
The sequence below works on my Ubuntu 16 installation. 100%. I did it a few times because I wanted to make sure I had done it correctly and repeatedly.
#mysql: reset root user password bash commands
sudo service mysql stop
sudo mkdir /var/run/mysqld
sudo chown mysql: /var/run/mysqld
sudo mysqld_safe --skip-grant-tables --skip-networking &
mysql -uroot mysql
#in msyql:
UPDATE mysql.user
SET
authentication_string=PASSWORD('PUT_NEW_PASSWORD_HERE'),
plugin='mysql_native_password'
WHERE User='root' AND Host='localhost';
exit;
#and back in bash
sudo mysqladmin -S /var/run/mysqld/mysqld.sock shutdown
sudo service mysql start
mysql -uroot -pPUT_NEW_PASSWORD_HERE
(Of course, mysql will beef that you put your password in the command line. Don't do it if your bash_history or logs could be accessed.)
By the way, I got this information from this website. Obviously this person is a genius. Props.
https://coderwall.com/p/j9btlg/reset-the-mysql-5-7-root-password-in-ubuntu-16-04-lts
So, I saw the PDF John posted discussing JSON format ideas. Ian and
Jon, you rock. It is a great document and excellent ideas. Most of it
makes good sense to me and, I'm sure that, as I reread and understand
better, I will love even more.
That said, there is one fundamental detail I do not love:
authors: {
'#contains': 'author',
'#items': [{ '#value': 'John Smith' }, { '#value': 'Dave Jones' }]
}
First
reason is that the label 'authors' is plural but contains only one
thing. In my opinion, things named plural should always be arrays.
Second is that I envision a line of code like:
const firstAuthor = inData.authors["#item"][0]["#value"];
Looks a lot like C# to me, low signal to noise ratio.
In
my other Javascript life, we would be inclined to use inflection, i.e.,
the assumption that 'authors' has elements with an implied name of
'author' and vice versa. I can understand that our XML roots make this
difficult to accept.
Consequently, I am inclined toward the everything is an object (if it's not a list) approach. EG,
authors: [
{ author: 'John Smith', '@type': 'bigshot' },
{ author: 'Dave Jones', '@type': 'contributor' }
]
This
provides a data structure that mentions the word 'author' the same
number of times as does the XML. That it also provides room for
attributes is good. This seems nicer to me:
const firstAuthor = inData.authors[0].author;
I
don't know if this can be expressed properly with openAPI or if it
violates some other rule of interaction with XML. I do know that, as a
Javascript programmer, I would rather use the form I suggest.
Trump's FCC is about to permanently turn the internet over to the corporations. In a few years, your ISP will be like a cable provider. You can only access the sites that make them money. Other sites will be very slow or non-existent. You will pay for packages. Package A: YouTube, Netflix. Package B: Netflix and Hulu. You want to start an internet business, you will have to work a deal with every ISP corporation. It will be bad.
Go to GoFccYourself.com to be forwarded to the correct page for your comment. Do it every day.
When you click on GoFccYourself.com, you will end up at a page that looks like this:
Click on the 'Express' link. It will take you to the entry form.
Suggested text:
Net neutrality is essential for freedom. Net neutrality requires Title II regulation of internet service providers. ISPs should completely prevented from influencing the cost or performance of the internet resources and websites I want to use. My bandwidth purchase from my ISP and the site's bandwidth purchase from their ISP should be the only charges.
Google as I might, nobody would tell me that the adapter had firmware that could be out of date. Eventually, I found a reference to the idea in the form of an updater that would not work.
The important thing is that I realized that if you cannot use your Macbook for presentations on a VGA projector or display, it might be that you simply need to buy a new one.
I did and now it works.
defaults write com.apple.dock persistent-apps -array-add '{"tile-type"="spacer-tile";}’;killall Dock;
You have to have NodeJS. If you don't have it, google it and make it happen.
Then you need to install prettier. It's NPM page is here.
To install it, type...
npm install prettier -g
This will install it in a command line utility.
In some file (I do a lot of this, so I created a Scripts folder and called the file, ~/Scripts/bin/js/runPrettier.js, you can do what works for you just remember that the bash file below has to point to the file), insert this:
#!/usr/local/bin/node
const prettier=require('prettier');
var inString='';
var writeStuff = function() {
var outString='';
outString=inString;
outString=outString.replace(/^[\s]$/gm, "/*linebreak*/"); //I like to retain linebreaks
outString=prettier.format(outString)
outString=outString.replace(/\/\*linebreak\*\//gm, ""); //you can remove these if you don't
process.stdout.write(outString);
};
//the rest ========================================================
process.stdin.resume();
process.stdin.setEncoding('utf8');
process.stdin.on('data', function(data){
inString+=data;
});
process.stdin.on('end', writeStuff);
Then in a file in the bbedit directory (~/Library/Application Support/BBEdit/Text Filters)
Create a file (I called this 'runPrettier') containing this...#!/bin/bash
~/Scripts/bin/js/runPrettier.js
chmod +x ~/Library/Application Support/BBEdit/Text Filters/runPrettier
I don't understand why it fails even when the server name matches something. However, if you have two separate servers with:
server_name xxx.com
server_name yyy.com
You would expect that (assuming that the configs appear in this order) that http://yyy.com would match that server_name. It will not. It will match xxx.com. Why? Because when there is not default, it simply uses the first server. Period.
If you have a default, though...
server_name xxx.com
server_name yyy.com
default_server
It works. yyy.com will match xxx.com.
I come upon this problem because I had a configuration that had the default file that comes in the distribution and it worked.
Then I added SSL. It did not work. Having long forgotten the issue with default, I debugged like a madman. Then I thought about the default issue (I ran into it sometime in the dark past - it is buried in the docs) and saw, There's a default right there!!!
Eventually (I know. This is the least entertaining punchline in history.), I realized that there was no default for port 443. QED
# Default server configuration
#
server {
listen 80 default_server;
listen [::]:80 default_server;
root /var/www/html;
index index.html index.htm index.nginx-debian.html;
server_name _;
location / {
try_files $uri $uri/ =404;
}
}
server {
listen 443 ssl default_server;
listen [::]:443 ssl default_server;
ssl on;
ssl_certificate /etc/ssl/PATH/TO/CERT.cer;
ssl_certificate_key /etc/ssl/PATH/TO/CERT.key;
root /var/www/html;
index index.html index.htm index.nginx-debian.html;
server_name _;
location / {
try_files $uri $uri/ =404;
}
}